对比Php和Ruby的getter/setter实现方式

作者:老王

一  反射的使用: 复制代码 代码如下:

Getter/Setter的用法在Java社区里很常见,比如说在Entity
Bean或者DTO中,这东西有的时候很必要,有的时候则乏味得很。Php社区一般都是跟着Java社区的步伐匍匐前进,所以很多人在思想上继承了这样的做法。

class Person{
 public $name;
 function __construct($name){
  $this->name=$name;
 }
}
interface Module{
 function execute();
}
class FtpModule implements Module{
 function setHost($host){
  print “FtpModule::setHost():$hostn”;
 }
 function setUser($user){
  print “FtpModule::setUser():$usern”;
 }
 function execute(){
  //something
 }
}
class PersonModule implements Module{
 function setPerson(Person $person){
  print “PersonModule::setPerson:{$person->name}n”;
 }
 function execute(){
  //something
 }
}
class ModuleRunner{
 private $configData
        =array(
          “PersonModule”=>array(‘person’=>’bob’),
         
“FtpModule”=>array(‘host’=>’example.com’,’user’=>’anon’)
        );
 private $modules=array();
 function init(){
  $interface=new ReflectionClass(‘Module’);
  foreach($this->configData as $modulename=>$params){
   $module_class=new
ReflectionClass($modulename);//根据配置configData的名称,实例化ReflectionClass
   if(!$module_class->isSubclassOf($interface)){//检查反射得到了类是否是$interface的子类
    throw new Exception(“unknown module
type:$modulename”);//不是Module子类则抛出异常
   }
   $module=$module_class->newInstance();//实例化一个FtpModule或者PersonModule对象
   foreach($module_class->getMethods() as
$method){//获得类中的方法
    $this->handleMethod($module,$method,$params);
   }
   array_push($this->modules,$module);//将实例化的module对象放入$modules数组中
  }
 }
 function handleMethod(Module $module,ReflectionMethod
$method,$params){
  $name=$method->getName();//获得方法名称
  $args=$method->getParameters();//获得方法中的参数
  if(count($args)!=1||substr($name,0,3)!=”set”){//检查方法必须是以set开头,且只有一个参数
   return false;
  }
  $property=strtolower(substr($name,3));//讲方法名去掉set三个字母,作为参数
  if(!isset($params[$property])){//如果$params数组不包含某个属性,就返回false
   return false;
  }
  $arg_class=@$args[0]->getClass;//检查setter方法的第一个参数(且唯一)的数据类型
  if(empty($arg_class)){
   $method->invoke($module,$params[$property]);
  }else{
   $method->invoke($module,$arg_class->newInstance($params[$property]));
  }
 }
}
$test=new ModuleRunner();
$test->init();
?>

先看看PHP中最一般的做法:

二  通过反射获得类中信息: 复制代码 代码如下:

class Demo { private $name; private $age;

class ReflectionUtil{
 static function getClassSource(ReflectionClass $class){
  $path=$class->getFileName();
  $lines=@file($path);
  $from=$class->getStartLine();
  $to=$class->getEndLine();
  $len=$to-$from+1;
  return implode(array_slice($lines,$from-1,$len));
 }
}
$classname="Person";
$path="../practice/{$classname}.php";
if(!file_exists($path)){
  throw new Exception("No such file as {$path}");
}
require_once($path);
if(!class_exists($classname)){
 throw new Exception("No such class as {$classname}");
}
print ReflectionUtil::getClassSource(new ReflectionClass('Person'));
?>

public function getName() { return $this-name; }

结果是:class Person{ public $age; public $name; function
getName(){return “zjx”;} function getAge(){return 12;} function
__toString(){ $rs=$this->getName();
$rs.=”(age”.$this->getAge().”)”; return $rs; } }

public function setName($name) { $this-name = $name; }

public function getAge() { return $this-$age; }

public function setAge($age) { $this-age = $age; }}

当然,因为Php脚本语言的灵活性,借助__call这样的魔术方法,如果你的属性很多,我们可以写得更简便些:

class Demo { private $name; private $age;

public function __call($name, $args) { if
(preg_match(”/^(get|set)(w+)/”, $name, $match)) { $firstChar =
substr($match[2], 0, 1); $lastChars = substr($match[2], 1);
$this-$attribute = strtolower($firstChar) . $lastChars;

if (”get” == $match[1]) { return $this-$attribute; } else {
$this-$attribute = $args[0];&

You can leave a response, or trackback from your own site.

Leave a Reply

网站地图xml地图