PHP const/static变量在父类的静态上下文中不可用

MPV*_*MPV 3 php oop inheritance class constants

由于某种原因(哪个?),子类中定义的PHP const/static变量在父类的静态上下文中不可用.

为什么?

例1:

class Model{
  function getAll(){
    $query = "SELECT * FROM " . self::DATABASE_TABLE_NAME;
    // ...
  }
}

class Post extends Model{
  const DATABASE_TABLE_NAME = 'post';
}

$p = Post::getAll();
Run Code Online (Sandbox Code Playgroud)

当我跑步时,我得到:

Fatal error: Undefined class constant 'DATABASE_TABLE_NAME' on line 3 
Run Code Online (Sandbox Code Playgroud)

($ query = ...的行)

例2:

class Model{
    function getAll(){
        $query = "SELECT * FROM " . self::$DATABASE_TABLE_NAME;
        // ...
    }
}

class Post extends Model{
    static $DATABASE_TABLE_NAME = 'post';
}

$p = Post::getAll();
Run Code Online (Sandbox Code Playgroud)

然后我得到:

Fatal error: Access to undeclared static property: Model::$DATABASE_TABLE_NAME on line 3
Run Code Online (Sandbox Code Playgroud)

(同一行)

Cro*_*zin 5

PHP5.3引入了后期静态绑定 - 这就是你要找的东西.

class ParentClass {
    public function getAll() {
        var_dump('Get all from ' . static::TABLE_NAME);
    }
}

class ChildClass extends ParentClass {
    const TABLE_NAME = 'my_table_name';
}

$c = new ChildClass();
$c->getAll(); // Get all from my_table_name
Run Code Online (Sandbox Code Playgroud)

编辑:

但是你应该设计你的类有点不同.上面的解决方案依赖于语言动态(你可以引用一些甚至不存在的东西(例如一个类常量)).在这样一个简单的例子中,一切都很好,但在实际情况下,这会导致生成可怕且难以维护的代码.

强制交付的class(ChildClass)实现一些返回表名的方法会更好:

abstract class ParentClass {
   // getAll function

   abstract protected function getTableName();
}

class ChildClass extends ParentClass {
    // You have to implement this method
    protected function getTableName() {
        return 'table name';
    }
}
Run Code Online (Sandbox Code Playgroud)