我有一个父对象,我在我的应用程序中用于一般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
编辑:应该说“私人”不受保护的儿童财产。
答案 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) "" }