2

我遇到了一个问题,我不确定这是否只是正常行为,或者我写错了什么。我的基类中有一个方法,它通过为该特定类的所有新实例创建代理来将全局过滤器应用于给定类。我计划的方式如下:

  1. 附加static $global_filter(代理)到我要过滤的类,它扩展了基类object
  2. 通过我的加载机制,在新实例化时返回代理而不是实际类(这将屏蔽方法调用并相应地应用过滤器)

但是,我陷入了第 1 步,似乎当我尝试分配static $global_filter给我想要过滤的后代类时,我的基类object也获得了相同的分配,这破坏了从它扩展的所有其他内容。

相关代码见下图:

class object {

    public static $global_filter;

    public function _filterGlobal($class, $method, $callback) {
        if ( !is_object($class::$global_filter) ) {
            $class::$global_filter = new filterable(null);
            # Replace the object being called with the new proxy.
        }
        var_dump($class);
        var_dump($class::$global_filter); // `filterable`
        var_dump(\core\blueprint\object::$global_filter); // Returns same as line above
        die();
        return $class::$global_filter->_add($method, $callback);
    }

}

$class::$global_filter\core\blueprint\object::$global_filter(基类)都返回相同的实例。而我希望object::$global_filter是 null

我没有使用后期静态绑定来保持一致性(单对象过滤器和全局过滤器都以相同的方式非静态地调用)。

这个问题似乎相关

任何帮助都感激不尽 :)

编辑,完整示例

这将是一个具体的类,它 extends modelwhich extendsobject

<?php
use core\blueprint\model;
class modelMock extends model {
    protected $schema = array();
    public function method($test) {
        return $test;
    }
}

这将是另一个对象(例如控制器),它也会扩展object。它将过滤器应用于所有新实例model

<?php
use core\blueprint\object;
class objectMock extends object {

    public function applyFilters() {
        $this->_filterGlobal('core\blueprint\model', 'method', function($self, $chain) {
            $chain->params[0] = 'new param'; // adjust the paramters
            return $chain->next();
        });
    }

}
4

3 回答 3

1

当我尝试将静态 $global_filter 分配给要过滤的后代类时,我的基类对象也得到相同的分配

是的,确实发生了这种情况。静态属性本质上是一个全局变量,被限制在类的命名空间内。遇到全局变量问题通常表明您没有使用最佳解决方案。

为了解决您的问题,您可以将过滤器设为(非静态)属性:

$class->$filter = new Whatever();

但与往常一样,通往罗马的道路更多,我建议您寻找替代方法。

于 2012-11-19T11:51:22.267 回答
1

我不知道这是否对你有帮助:

class a {
   public static $type;

   public static function setType($class, $newType) {
      $class::$type = $newType;
      var_dump($class::$type);
   }
}
class b {
   public static $type = 'myType';
}

var_dump(b::$type);
a::setType('b', 'yourType');
var_dump(a::$type);

可能是您没有为具体类定义静态属性。

于 2012-11-19T12:06:16.583 回答
1

感谢大家的帮助,今天早上我花了一些时间并设法解决了我的问题。这是一个解决方法,但它是这样的:

public function _filterGlobal($class, $method, $callback) {
    if ( !is_object($class::$global_filter[$class]) ) {
        $class::$global_filter[$class] = new filterable(null);
        # Replace the object being called with the new proxy.
    }
    return $class::$global_filter[$class]->_add($method, $callback);
}

因此,基本上为了在子类中获得唯一的静态变量而不必显式定义它们,您可以使用一个数组来存储子类的名称作为键,然后通过 getter 访问这些变量。

于 2012-11-19T21:09:32.613 回答