在php中,如何在涉及对象代理时确定相等性?

时间:2013-10-06 00:39:59

标签: php design-patterns proxy

在我的php应用程序中,我一直在将对象与通常的相等比较运算符进行比较,例如:

if ($objectA == $objectB) { ... }

最近我实现了代理(对于加载昂贵的对象),但这意味着相等运算符不再起作用。有一个简单的方法吗?一个不依赖于反思的人?

目前,我已经尝试测试每个对象的唯一标识符,例如

if ($objectA->getId() == $objectB->getId) { ... }

但是这有两个问题:1)我需要重构所有现有代码,2)将来我可能需要比较值对象(而不是实体)的对象。

我不希望一个简单的解决方案,因为我认为它需要一种新的魔术方法......

这是我的AbstractProxy类。任何帮助表示赞赏...

abstract class KOOP_Base_AbstractProxy
    implements KOOP_Base_iDomain
{
    use KOOP_Trait_Helper_Helper;

    /**
     * @var integer Object identifier
     */
    protected $_id = null;

    /**
     * @var KOOP_Base_AbstractMapper
     */
    protected $_mapper = null;

    /**
     * @var KOOP_Base_AbstractDomain Actual object
     */
    protected $_subject = null;

    /**
     * Store object id for lazy loading
     *
     * @param integer   $id Object identifier
     * @param string    $mapper Mapper by which to retrieve object
     */
    public function __construct($id, $mapper)
    {
        $this->_id = $id;
        $this->_mapper = $mapper;
    }

    /**
     * Get subject
     *
     * @return KOOP_Base_AbstractDomain
     */
    protected function getSubject()
    {
        if (!$this->_subject) {
            $this->_subject = $this->getMapper($this->_mapper)->find($this->_id);
        }
        return $this->_subject;
    }

    /**
     * Get property
     *
     * @param string $property
     * @return mixed
     */
    public function __get($property)
    {
        return $this->getSubject()->$property;
    }

    /**
     * Set property
     *
     * @param string $property
     * @param mixed $value
     * @return void
     */
    public function __set($property, $value)
    {
        $this->getSubject()->$property = $value;
    }

    /**
     * Is property set?
     *
     * @param $property
     * @return boolean
     */
    public function __isset($property)
    {
        return isset($this->getSubject()->$property);
    }

    /**
     * Unset property
     *
     * @param string $property
     * @return mixed
     */
    public function __unset($property)
    {
        unset($this->getSubject()->$property);
    }

    /**
     * Call method
     *
     * @param   string $method Method to call
     * @param   array  $params Parameters to pass
     * @return  mixed
     */
    public function __call($method, array $params)
    {
        return call_user_func_array(array($this->getSubject(), $method), $params);
    }

    /**
     * Get id
     *
     * Saves having to retrieve the entire object when only the ID is required.
     */
    public function getId()
    {
        return $this->_id;
    }
}

1 个答案:

答案 0 :(得分:1)

代理确实打破了对象的平等,并且没有完全清晰的方法来解决这个问题。在完全面向对象的语言中,您可以通过运算符重载(我不建议)或实现自定义.equals()函数来处理它(如在Java中)。遗憾的是,PHP根本不支持此级别的面向对象,因此您将做出一些决定。

1)我更希望你的代理类提供一个equals()函数,它接受你要测试的对象的引用作为输入,并将它与代理对象进行比较 - 这不应该更贵' '而不是根本不使用代理。伪PHP代码中的示例(如果我的参考语法已关闭,我很抱歉,已经有一段时间了):

public function equals (&$toCompare)
{

 if ($_subject == $toCompare)
 {
    return true;
 }
 else
 {
    return false;
 }

}

缺点很简单:您必须重构涉及此代理对象的代码,并且您必须记住“==”在您工作时对此代理对象类型不起作用。如果你不太多地处理这些对象,或者如果你一直处理它们,这很好。如果你经常但间歇性地处理它们,或者其他人偶尔必须与它们一起工作,那么当你/他们忘记这个平等问题时,这将导致错误。

2)使用Operator Overloading extension语言。我没有这样做,我不知道它是否有效,这可能是一场噩梦。我把它包括在理论上完整性。

就我个人而言,我认为我只是使用伪Java方法将其称为一天,因为我认为它实际上是可行的,只需要正确使用该函数(并记住首先使用它) )。