Ошибка сериализации дерева объектов с помощью SplObjectStorage - PullRequest
5 голосов
/ 27 августа 2010

Я реализовал простой шаблон Composite, используя SplObjectStorage, как в примере выше:

class Node
{
    private $parent = null;

    public function setParent(Composite $parent)
    {
        $this->parent = $parent;
    }
}

class Composite extends Node
{
    private $children;

    public function __construct()
    {
        $this->children = new SplObjectStorage;
    }

    public function add(Node $node)
    {
        $this->children->attach($node);
        $node->setParent($this);
    }
}

Всякий раз, когда я пытаюсь сериализовать объект Composite, PHP 5.3.2 бросает мне Segmentation Fault. Это происходит только тогда, когда я добавляю к объекту любое количество узлов любого типа.

Это код ошибки:

$node = new Node;
$composite = new Composite;
$composite->add($node);
echo serialize($composite);

Хотя этот работает:

$node = new Node;
$composite = new Composite;
echo serialize($composite);

Также, если я реализую шаблон Composite с помощью array () вместо SplObjectStorage, все тоже будет работать нормально.

Что я делаю не так?

1 Ответ

8 голосов
/ 27 августа 2010

Устанавливая Родителя, вы получаете круговую ссылку. PHP будет пытаться сериализовать композит, все его узлы и узлы, в свою очередь, будут пытаться сериализовать композит ..

Вы можете использовать магические методы __sleep и __wakeup(), чтобы удалить (или сделать что-либо другое) родительскую ссылку при сериализации.

EDIT:

Посмотрите, исправит ли это добавление к Composite:

public function __sleep()
{
    $this->children = iterator_to_array($this->children);
    return array('parent', 'children');
}
public function __wakeup()
{
    $storage = new SplObjectStorage;
    array_map(array($storage, 'attach'), $this->children);
    $this->children = $storage;
}
...