我正在开发一个框架,我试图尽可能地强输入。(我在 PHP 中工作,并从 C# 中获取我喜欢的一些想法,并尝试在这个框架中利用它们。)我正在创建一个 Collection 类,它是域实体/对象的集合。它有点模仿List<T>
.Net 中的对象。
我遇到了阻碍我输入这门课程的障碍。如果我有一个 UserCollection,它应该只允许用户对象进入它。如果我有 PostCollection,它应该只允许 Post 对象。
这个框架中的所有Collection都需要具备一定的基本功能,比如add、remove、iterate。我创建了一个界面,但发现我无法执行以下操作:
interface ICollection { public function add($obj) }
class PostCollection implements ICollection { public function add(Post $obj) {} }
这破坏了它对界面的遵守。但是我不能强类型化接口,因为所有集合都是同一类型。所以我尝试了以下方法:
interface ICollection { public function add($obj) }
abstract class Collection implements ICollection { const type = 'null'; }
class PostCollection extends Collection {
const type = 'Post';
public function add($obj) {
if(!($obj instanceof self::type)) {
throw new UhOhException();
}
}
}
当我尝试运行这段代码时,我得到syntax error, unexpected T_STRING, expecting T_VARIABLE or '$'
了instanceof
声明。对该问题进行一些研究,看起来原因的根源是$obj instanceof self
对类进行测试是有效的。PHP 似乎没有处理self::type
表达式中的整个常量语句。在变量周围添加括号会self::type
引发有关意外“(”的错误。
一个明显的解决方法是不要使type
变量成为常量。该表达式$obj instanceof $this->type
工作得很好($type
当然,如果被声明为变量)。
我希望有一种方法可以避免这种情况,因为我想将该值定义为一个常量,以避免以后对变量进行任何可能的更改。关于如何实现这一点的任何想法,或者在这方面我是否将 PHP 发挥到了极限?有没有一种“转义”或封装的方法,self::this
以便 PHP 在处理时不会死?
更新根据下面的反馈,我想到了一些尝试——下面的代码有效!谁能想到 1) 不这样做的理由,2) 这最终不会奏效的原因,或 3) 更好的方法来实现这一目标?
interface ICollection { public function add($obj) }
abstract class Collection { const type = null; protected $type = self::type; }
class PostCollection extends Collection {
const type = 'Post';
public function add($obj) {
if(!($obj instanceof $this->type)) {
throw new UhOhException();
}
}
}
更新#2:将上面的代码投入生产后,结果证明它不起作用。当我测试它时,我不知道它是如何工作的,但它根本不起作用。我想我坚持使用protected
变量。