PHP在父级中访问子级的私有属性

时间:2010-08-14 16:26:12

标签: php oop serialization protected

我有一个父对象,我在我的应用程序中用于一般CRUD - 它具有基本保存和放大器功能。检索方法,所以我不必将它们重新包含在我的所有对象中。我的大多数子对象都扩展了这个基础对象。这工作正常,但我发现检索序列化子对象时出现问题。我在父对象中使用“retrieve”方法创建子实例,然后从未序列化的子属性填充自己 - 这意味着可以“自行反序列化”对象。

唯一的问题是 - 如果子对象具有protected或private属性,则父对象无法读取它,因此在检索期间不会被拾取。

因此,我正在寻找一种更好的方法来“自我反序列化”或允许父对象“看到”受保护属性的方法 - 但仅在检索过程中。

代码示例:

BaseObject {

 protected $someparentProperty;

 public function retrieve() {

  $serialized = file_get_contents(SOME_FILENAME);
  $temp = unserialize($serialized);
  foreach($temp as $propertyName => $propertyValue) {
    $this->$propertyName = $propertyValue;
  }     

 }

 public function save() {

    file_put_contents(SOME_FILENAME, serialize($this));
 }
}

class ChildObject extends BaseObject {

 private $unretrievableProperty;  

 public setProp($val) {
    $this->unretrivableProperty = $val;
 }
}

$tester = new ChildObject();
$tester->setProp("test");
$tester->save();

$cleanTester = new ChildObject();
$cleanTester->retrieve();
// $cleanTester->unretrievableProperty will not be set

编辑:应该说“私人”不受保护的儿童财产。

4 个答案:

答案 0 :(得分:3)

尝试这样:

abstract class ParentClass
{
  protected abstract function GetChildProperty();

  public function main()
  {
    $value = $this->GetChildProperty();
  }
}

class ChildClass extends ParentClass
{
  private $priv_prop = "somevalue";

  protected function GetChildProperty()
  {
    return $this->priv_prop;
  }
}

答案 1 :(得分:0)

如果子对象中的getProperty()函数返回$ this-> unretrievableProperty

答案 2 :(得分:0)

似乎相同的类可见性策略不适用于iherited / parent类。 php文档没有解决这个问题。

我建议您将retrieve方法声明为static,并通过静态调用而不是当前的“self unserialize”方法获取$ cleanTester。

static function retrieve() {
  $serialized = file_get_contents(SOME_FILENAME);
  return unserialize($serialized);
}

[...]

$cleanTester = BaseObject::retrieve();

或者您可以使用__get()方法访问不可访问的属性...我相信这可以添加到BaseObject类并从子类中获取受保护的属性。由于同一类可见性政策应适用于BaseObject,因此您可以将__get()方法定义为私有或受保护。

BaseObject {
  private function __get($propertyName) {
    if(property_exists($this,$propertyName))
      return $this->{$propertyName};

    return null;
  }

答案 3 :(得分:0)

解决此问题的最佳方法是使用反射

例如:

$_SESSION[''] = ''; // init

class Base {
    public function set_proxy(){
        $reflectionClass = new ReflectionClass($this);
        $ar = $reflectionClass->getDefaultProperties();

        !isset($ar['host'])  or  $_SESSION['host'] = $ar['host'];
    }
}

class Son1 extends Base {
    private $host = '2.2.2.2';
}

class Son2 extends Son1 {

}


$son1 = new Son1();
$son1->set_proxy(); 
var_dump($_SESSION); // array(2) { [""]=> string(0) "" ["host"]=> string(7) "2.2.2.2" }

unset($_SESSION);
$_SESSION[''] = ''; // init

$son2 = new Son2();
$son2->set_proxy(); 
var_dump($_SESSION); //  array(1) { [""]=> string(0) "" }