Problem: I am trying to extend PHP’s ArrayObject as shown below. Unfortunately I can’t get it to work properly when setting multi-dimensional objects and instead an error thrown as I have the strict settings enabled in PHP. (Error: Strict standards: Creating default object from empty value)
Question: How can I modify my class to automatically create non-existing levels for me?
The code:
$config = new Config;
$config->lvl1_0 = true; // Works
$config->lvl1_1->lvl2 = true; // Throws error as "lvl1" isn't set already
class Config extends ArrayObject
{
function __construct() {
parent::__construct(array(), self::ARRAY_AS_PROPS);
}
public function offsetSet($k, $v) {
$v = is_array($v) ? new self($v) : $v;
return parent::offsetSet($k, $v);
}
}
Taking a more oop view of your issue, you can create a class that models the concept of an multi-dimensional object.
The solution im posting doesn’t extends from
ArrayObjectto achieve the goals you mention. As you tagged your question as oop, i think it´s important to reinforce the separation the way you store an object’s state from how do you access it.Hope this will help you achieve what you need!
From what you said, an multi-dimensional object is one that:
$config->database->host = 'localhost'thedatabaseandhostlevels are initialized automatically, andhostwill return'localhost'when queried.Proposed Solution
So, how can those features be implemented?
The second one is easy: using PHP’s
__getand__setmethods. Those will get called whenever an read/write is beign done on an inaccessible property (one that’s not defined in an object).The trick will be then not to declare any property and handle propertie’s operations through those methods and map the property name being accessed as a key to an associative array used as storage. They’ll provide basically an interface for accessing information stored internally.
For the third one, we need a way to create a new nesting level when a undeclared property is read.
The key point here is realizing that the returned value for the property must be an multi-dimensional object so further levels of nesting can be created from it also: whenever we´re asked for a property whose name is not present in the internal array, we´ll associate that name with a new instance of
MultiDimensionalObjectand return it. The returned object will be able to handle defined or undefined properties too.When an undeclared property is written, all we have to do is assign it’s name with the value provided in the internal array.
The fourth one is easy (see it on
__constructimplementation). We just have to make sure that we create anMultiDimensionalObjectwhen a property’s value is an array.Finally, the fist one: the way we handle the second and third features allows us to read and write properties (declared and undeclared) in any level of nesting.
You can do things like
$config->foo->bar->baz = 'hello'on an empty instance and then query for$config->foo->bar->bazsuccessfully.Important
Notice that
MultiDimensionalObjectinstead of beign itself an array is it composed with an array, letting you change the way you store the object’s state as needed.Implementation
Demo
Code:
Result:
Code:
Result:
Code:
Result:
Code:
Result: