如果我有一个像这样的基本抽象类:
<?php
abstract class Record {
static $table;
public function getRows () {
return getRowsFromTable(static::$table);
}
}
?>
我想像这样扩展这个类:
<?php
class User extends Record {
static $table = 'users';
private $name;
?>
然后,如果我打电话:
<?php
$user = new User;
$user->getRows();
?>
在内部,getRows()
呼叫并返回getRowsFromTable('users')
。
但是,如果我要创建另一个也扩展Record
的类:
<?php
class House extends Record {
static $table = 'houses';
private $address;
?>
然后,该static $table = 'houses';
声明将重写Record::$table
,从而破坏User
类。
正在发生的事情是,声明static $table = 'houses';
冒泡到父类,所以现在Record::$table = 'houses';
.由于House
是在User
之后声明的,下次我在内部调用$user->getRows()
时,User
引用父Record
,最终调用getRowsFromTable('houses')
而不是getRowsFromTable('users')
。
我正在使用后期静态绑定,以便从扩展类中获取属性;但由于User
和House
扩展相同的父类,因此它们最终都具有相同的属性值,尽管它们使用不同的值覆盖它。
如果我通过创建一个类Record2
并House
扩展Record2
来复制Record
类,我就不会有这个问题——但这并没有真正的帮助。
这是错误的设置吗?我不应该在这种情况下使用静态变量吗?如果是这样,我应该在他们的位置放什么?我知道$table
不一定是静态的,但还有其他属性可能需要是静态的。
我真的不会为此目的使用静态字符串。
像这样的事情怎么样...
abstract class Record {
/**
* @return table name as string
*/
abstract protected function getTable();
public function getRows () {
return getRowsFromTable($this->getTable());
}
}
然后,您的具体实现必须实现getTable
,例如
class User extends Record {
protected function getTable() {
return 'users';
}
}