tags:

views:

766

answers:

8

I'm going to try something with the format of this question and I'm very open to suggestions about a better way to handle it.

I didn't want to just dump a bunch of code in the question so I've posted the code for the class on refactormycode.

base-class-for-easy-class-property-handling

My thought was that people can either post code snippets here or make changes on refactormycode and post links back to their refactorings. I'll make upvotes and accept an answer (assuming there's a clear "winner") based on that.

At any rate, on to the class itself:

I see a lot of debate about getter/setter class methods and is it better to just access simple property variables directly or should every class have explicit get/set methods defined, blah blah blah. I like the idea of having explicit methods in case you have to add more logic later. Then you don't have to modify any code that uses the class. However I hate having a million functions that look like this:

public function getFirstName()
{
   return $this->firstName;
}
public function setFirstName($firstName)
{
   return $this->firstName;
}

Now I'm sure I'm not the first person to do this (I'm hoping that there's a better way of doing it that someone can suggest to me).

Basically, the PropertyHandler class has a __call magic method. Any methods that come through __call that start with "get" or "set" are then routed to functions that set or retrieve values into an associative array. The key into the array is the name of the calling method after get or set. So, if the method coming into __call is "getFirstName", the array key is "FirstName".

I liked using __call because it will automatically take care of the case where the subclass already has a "getFirstName" method defined. My impression (and I may be wrong) is that the __get & __set magic methods don't do that.

So here's an example of how it would work:

class PropTest extends PropertyHandler
{
    public function __construct()
    {
        parent::__construct();
    }
}

$props = new PropTest();

$props->setFirstName("Mark");
echo $props->getFirstName();

Notice that PropTest doesn't actually have "setFirstName" or "getFirstName" methods and neither does PropertyHandler. All that's doing is manipulating array values.

The other case would be where your subclass is already extending something else. Since you can't have true multiple inheritance in PHP, you can make your subclass have a PropertyHandler instance as a private variable. You have to add one more function but then things behave in exactly the same way.

class PropTest2
{
    private $props;

    public function __construct()
    {
        $this->props = new PropertyHandler();
    }

    public function __call($method, $arguments)
    {
        return $this->props->__call($method, $arguments);
    }
}

$props2 = new PropTest2();

$props2->setFirstName('Mark');
echo $props2->getFirstName();

Notice how the subclass has a __call method that just passes everything along to the PropertyHandler __call method.


Another good argument against handling getters and setters this way is that it makes it really hard to document.

In fact, it's basically impossible to use any sort of document generation tool since the explicit methods to be don't documented don't exist.

I've pretty much abandoned this approach for now. It was an interesting learning exercise but I think it sacrifices too much clarity.

+3  A: 

The way I do it is the following:

class test {
 protected $x='';
 protected $y='';

 function set_y ($y) {
  print "specific function set_y\n";
  $this->y = $y;
 }

 function __call($function , $args) {
  print "generic function $function\n";
  list ($name , $var ) = split ('_' , $function );
  if ($name == 'get' && isset($this->$var)) {   return $this->$var;  }
  if ($name == 'set' && isset($this->$var)) { $this->$var= $args[0];  return;}
  trigger_error ("Fatal error: Call to undefined method test::$function()");
 }
}

$p = new test();
$p->set_x(20);
$p->set_y(30);
print $p->get_x();
print $p->get_y();

$p->set_z(40);

Which will output (line breaks added for clarity)

generic function set_x
specific function set_y

generic function get_x
20
generic function get_y
30

generic function set_z
Notice: Fatal error: Call to undefined method set_z() in [...] on line 16
Pat
(moving comment from OP): @Pat That's interesting. Certainly much less code. But you still have to explicitly declare each underlying property variable, right?
Marc Gravell
+1  A: 

Yes that's right the variables have to be manually declared but i find that better since I fear a typo in the setter

$props2->setFristName('Mark');

will auto-generate a new property (FristName instead of FirstName) which will make debugging harder.

Pat
+1  A: 

I like having methods instead of just using public fields, as well, but my problem with PHP's default implementation (using __get() and __set()) or your custom implementation is that you aren't establishing getters and setters on a per-property basis. My problem with this is that adding "more logic later" requires that you add blanket logic that applies to all properties accessed with the getter/setter or that you use if or switch statements to evaluate which property you're accessing so that you can apply specific logic.

I like your solution, and I applaud you for it--I'm just not satisfied with the limitations that PHP has when it comes to implicit accessor methods.

Brian Warshaw
(moving comment from OP) @Brian That's an interesting point. I'm going to have to mull that over a bit.
Marc Gravell
+1  A: 

@Brian

My problem with this is that adding "more logic later" requires that you add blanket logic that applies to all properties accessed with the getter/setter or that you use if or switch statements to evaluate which property you're accessing so that you can apply specific logic.

That's not quite true. Take my first example:

class PropTest extends PropertyHandler
{
    public function __construct()
    {
        parent::__construct();
    }
}

$props = new PropTest();

$props->setFirstName("Mark");
echo $props->getFirstName();

Let's say that I need to add some logic for validating FirstNames. All I have to do is add a setFirstName method to my subclass and that method is automatically used instead.

class PropTest extends PropertyHandler
{
    public function __construct()
    {
        parent::__construct();
    }

    public function setFirstName($name)
    {
        if($name == 'Mark')
        {
            echo "I love you, Mark!";
        }
    }
}

I'm just not satisfied with the limitations that PHP has when it comes to implicit accessor methods.

I agree completely. I like the Python way of handling this (my implementation is just a clumsy rip-off of it).

Mark Biek
+1  A: 

@Mark

But even your method requires a fresh declaration of the method, and it somewhat takes away the advantage of putting it in a method so that you can add more logic, because to add more logic requires the old-fashioned declaration of the method, anyway. In its default state (which is where it is impressive in what it detects/does), your technique is offering no advantage (in PHP) over public fields. You're restricting access to the field but giving carte blanche through accessor methods that don't have any restrictions of their own. I'm not aware that unchecked explicit accessors offer any advantage over public fields in any language, but people can and should feel free to correct me if I'm wrong.

Brian Warshaw
A: 

I've always handled this issue in a similar with a __call which ends up pretty much as boiler plate code in many of my classes. However, it's compact, and uses the reflection classes to only add getters / setters for properties you have already set (won't add new ones). Simply adding the getter / setter explicitly will add more complex functionality. It expects to be

Code looks like this:

/**
* Handles default set and get calls
*/
public function __call($method, $params) {

 //did you call get or set
 if ( preg_match( "|^[gs]et([A-Z][\w]+)|", $method, $matches ) ) {

  //which var?
  $var = strtolower($matches[1]);

  $r = new ReflectionClass($this);
  $properties = $r->getdefaultProperties();

  //if it exists
  if ( array_key_exists($var,$properties) ) {
   //set
   if ( 's' == $method[0] ) {
    $this->$var = $params[0];
   }
   //get
   elseif ( 'g' == $method[0] ) {
    return $this->$var;
   }
  }
 }
}

Adding this to a class where you have declared default properties like:

class MyClass {
    public $myvar = null;
}

$test = new MyClass;
$test->setMyvar = "arapaho";

echo $test->getMyvar; //echos arapaho

The reflection class may add something of use to what you were proposing. Neat solution @Mark.

reefnet_alex
A: 

Just recently, I also thought about handling getters and setters the way you suggested (the second approach was my favorite, i.e. the private $props array), but I discarded it for it wouldn't have worked out in my app.

I am working on a rather large SoapServer-based application and the soap interface of PHP 5 injects the values that are transmitted via soap directly into the associated class, without bothering about existing or non-existing properties in the class.

Cassy
A: 

I can't help putting in my 2 cents...

I have taken to using __get and __set in this manor http://gist.github.com/351387 (similar to the way that doctrine does it), then only ever accessing the properties via the $obj->var in an outside of the class. That way you can override functionality as needed instead of making a huge __get or __set function, or overriding __get and __set in the child classes.

SeanJA