How to create data structure with key duplicates in php instead of default hash?
I want to create wrapper class, which will enable keys duplicates while default hash does not allow it. Class should use member overloading mechanism introduced in php5, so it would imitate all the behavior standard hash has. For example, I want to have smth like
$var => obj( :values_arr -> array(
开发者_StackOverflow社区 obj(:key -> 'mykey', :value -> 'val1'),
obj(:key -> 'mykey', :value -> 'val2')
)
)
If I want to get $var['mykey'], it should return array('val1', 'val2'), but if I want to extend obj with new 'mykey' => 'value' pair, I would call
$val['mykey'][] = 'value'
Main idea is that behavior of the hash was preserved and after attempt to assign value to using existing key, it wouldn't be overwritten, but appended to the list.
How would you imitate other data structures in php5 (before 5.3)? Are there any known solutions or examples you want to share?
like this
class MultiMap
{
protected $map = array();
function __set($key, $val) {
if(!isset($this->map[$key]))
return $this->map[$key] = $val;
if(!is_array($this->map[$key]))
$this->map[$key] = array($this->map[$key]);
$this->map[$key][] = $val;
}
function __get($key) {
return $this->map[$key];
}
}
$m = new MultiMap;
$m->foo = 1;
$m->foo = 2;
$m->bar = 'zzz';
print_r($m->foo);
print_r($m->bar);
but the whole idea looks a bit odd to me. Can you explain why you need this?
it's not clear for me why you need operators as keys in your AST perhaps a structure like this would be more convenient
('op' => 'AND', 'args' => [
(op => AND, args => [
(op => atom, value => word1),
(op => atom, value => word2),
]),
(op => AND, args => [
(op => atom, value => word3),
(op => atom, value => word4),
])
])
You can achieve the array syntax
$val['mykey'] = 'value';
with the ArrayAccess
interface
class MultiHash implements ArrayAccess, IteratorAggregate
{
protected $data;
public function offsetGet($offset)
{
return $this->data[$offset];
}
public function offsetSet($offset, $value)
{
if ($offset === null) { // $a[] = ...
$this->data[] = array($value);
} else {
$this->data[$offset][] = $value;
}
}
public function offsetExists($offset)
{
return isset($this->data[$offset]);
}
public function offsetUnset($offset)
{
unset($this->data[$offset]);
}
public function getIterator()
{
$it = new AppendIterator();
foreach ($this->data as $key => $values) {
$it->append(new ConstantKeyArrayIterator($values, 0, $key));
}
return $it;
}
}
class ConstantKeyArrayIterator extends ArrayIterator
{
protected $key;
public function __construct($array = array(), $flags = 0, $key = 0)
{
parent::__construct($array,$flags);
$this->key = $key;
}
public function key()
{
return parent::key() === null ? null : $this->key;
}
}
I also implemented IteratorAggregate
to allow iteration over all single elements.
Test Code
$test = new MultiHash();
$test[] = 'foo';
$test[] = 'bar';
$test['mykey'] = 'val1';
$test['mykey'] = 'val2';
$test['mykey2'] = 'val3';
echo "mykey: ";
var_dump($test['mykey']);
echo "mykey2: ";
var_dump($test['mykey2']);
echo "iterate:\n";
foreach ($test as $key => $value) {
echo "$key : $value \n";
}
Test Output
mykey: array(2) {
[0]=>
string(4) "val1"
[1]=>
string(4) "val2"
}
mykey2: array(1) {
[0]=>
string(4) "val3"
}
iterate:
0 : foo
1 : bar
mykey : val1
mykey : val2
mykey2 : val3
精彩评论