如何在PHP中获取对根对象的迭代对象引用?

时间:2015-02-25 22:25:27

标签: php oop design-patterns simplexml

我有一个类需要一些基本的HTML5,并且有一些DOM魔法将它变成一个扩展类为simpleXMLElement的类。这一切都始于"工厂" (我可能会略微滥用这个术语)叫做XPage的类。

因此我们可能

<?php
$MyPage = new XPage($HTML);

到目前为止一切顺利。我现在需要做的是选择XML文档的某些部分,以便它们可以被标记为&#34; (我确信有一个更好的名称)所以可以这样做

$MyPage->ThatBit("foo")->addChild('p',$MyParagraph);

理想情况下,我希望能够

$MyPage->page()->body->header->RememberThisBitAs("foo");

或者直观相似的东西。然后应该发生的是对象$ MyPage(类型为XPage)应该传递引用,在这种情况下是html / body / header(可能不是正确的XPath表示法,但希望你可以遵循)。

这个想法是,最常见的访问区域可以更顺利地获得。这就是SimpleXML类(即使使用我的扩展名)遇到的事情,如果没有其他工厂,也会使XPage成为单例。

然而,XPage类($ MyPage)包含&#34;书签&#34;的数组。据我所知,没有办法在SimpleXML的迭代中传递引用,以便孩子或孩子的孩子不能告诉工厂&#34;请给我收藏&#34;。

这会让我做出令人讨厌的事情:

$MyPage->addBookmark("foo", 
$MyPage->page()->body->header->getChildren() );

这似乎不对。

$MyPage->page()->body->header->RememberThisBitAs("foo",$MyPage);

这不是那么糟糕但仍然感觉&#34;错。

如果没有过多的对象耦合或丑陋的自引用,我该如何处理?

更新

我想知道是否可以使用debug_backtrace反向遍历一个类,但答案似乎是否。

In PHP is it possible to reverse traverse a Traversable class to find the root object?

其他类似的测试表明,在可遍历堆栈中的任何其他元素上都无法访问在元素上设置类变量的值。

我将专注于包装器建议。

更新

SimpleXML实际上不是真正的对象,而是系统资源。这可以解释我的痛苦来自哪里。说出笔记:http://php.net/manual/en/class.simplexmlelement.php

因此,对于所有对象而言,感觉就像一串对象互相调用,实际情况并非如此。

2 个答案:

答案 0 :(得分:4)

选项1:对根

的引用

您可以修改XPage类的方法以返回XPage对象而不是simpleXMLElements(或另一个扩展simpleXMLElement的类)。那个类可以有一个额外的属性,持有对&#34; root&#34;的引用。元件。

基本上你的&#34;包装类&#34;会看起来像这样(前面的伪代码!):

class Wrap extends simpleXMLElement
{
    private $root = null;
    private $refs = array();

    public function addReference($name, $element = null) {
        if($element === null) {
            $element = $this;
        }
        if($this->root === null) {
            $this->refs[$name] = $this;
        } else {
            $this->root->addReference($name, $this);
        }
    }

    public function getReference($name) {
        if($this->root === null) {
            return $this->refs[$name];
        } else {
            return $this->root->getReference($name);
        }
    }
}

所以上面的用例看起来像这样:

$MyPage->page()->body->header->addReference("foo");

$MyPage->getReference("foo")->addChild('p',$MyParagraph);

选项2:添加ID

或者,您可以只为该元素添加一个id属性,然后通过它来检索它:

$MyPage->page()->body->header->addAttribute("id", "foo");

$MyPage->xpath("//*[@id='foo']")->addChild('p',$MyParagraph);

答案 1 :(得分:0)

鉴于Lars Ebert的答案给予了极大的推动,而在考虑我遇到的一些疯狂(和不可行)的想法之前Extend Class with Final Constructor on PHP让我明白我需要的设计模式是代表团的设计模式类。

由于我可能没有将专业函数添加到SimpleXMLElement扩展,我使用魔术方法来创建委托包装器。缺点是这在我的IDE中不能很好地运行,但它在初始测试时完全正常工作。我已经拿出了与我自己的项目有关的东西,但另外这里是我正在使用的解决方案。

它需要XPage对象实现一些相同的接口元素(我可能会定义一个接口),它使用我的扩展和指向自身的指针将SimpleXML元素传递给这个委托包装器,它的行为类似于SimpleXML,但它确实是一个对象并且可以将调试消息和自引用传递回XPage。

<?php

/**
 * A Delegation extension of SimpleXMLElement
 * 
 * Designed to act like SimpleXML but builds a back traceable tree that expects
 * XPage (or some other parent factory) to catch references and debug messages.
 * 
 * @author Lord Matt 
 * 
 * @license
 *  Copyright (C) 2015 Matthew Brown
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2 of the License, or
 *  (at your option) any later version.
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 *  GNU General Public License for more details.
 *  You should have received a copy of the GNU General Public License along
 *  with this program; if not, write to the Free Software Foundation, Inc.,
 *  51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

class AEWrapper {

    protected $mother = null;
    protected $MyAE = null;

    public function __construct($MyAE,$mother) {
        $this->MyAE = $MyAE;
        $this->mother = $mother;
        return $this;
    }

    public function __call($name, $arguments) {
        try {
            $result = call_user_method_array($name, $this->MyAE, $arguments);
        } catch (Exception $e) {
            $this->log_error('AEWrapper caught exception: ',  $e->getMessage());
            return FALSE;
        }
        if(is_object($result)){
            return new AEWrapper($result,$this);
        }
        return $result;
    }

    public function __get($name) {
        try {
            $result = $this->MyAE->{$name};
        } catch (Exception $e) {
            $this->log_error('AEWrapper caught exception: ',  $e->getMessage());
            return FALSE;
        }
        if(is_object($result)){
            return new AEWrapper($result,$this);
        }
        return $result;
    }

    public function &actual_simpleXML_obj_please(){
        return $this->MyAE;
    }

    // methods exists in XPage
    public function register_as($name,$what=null){
        if($what===null){
            $what = $this;
        }
        $this->mother->register_as($name,$what);
    }

    // [SNIP] 
    // Other project specific stuff here
    // [/SNIP]

    // methods exists in XPage    
    protected function log_error($error){
        $this->mother->log_error('/'.$error);
    }
}