Как переопределить статическое свойство родительского объекта и позволить родительскому объекту получить доступ к новому значению в PHP?

Это то, что у меня есть: все объекты, которые могут сохраняться в базе данных, расширяют абстрактный класс DatabaseObject, который имеет весь логический код, который фактически отслеживает изменения атрибутов и запускает запросы к базам данных.

Я использую две статические переменные для определения конкретных объектов. Я определяю их в основном в базовом классе, а затем, предположительно, переписываю их в реальных объектах базы данных.

Проблема заключается в том, что когда код в родительском классе фактически выполняется, он использует старое родительское значение вместо текущего значения объекта.

Здесь код базового класса:

abstract class DatabaseObject {

    public $id;

    private static $databaseTable = NULL;
    private static $databaseFields = array();

    private $data = array();
    private $changedFields = array();

    public function IDatabaseObject($id) {
    $this->id = $id;
    $this->data = Database::GetSingle(self::$databaseTable, $id);

    Utils::copyToObject($this, $this->data, self::$databaseFields);
    }

    public static function Load($id) {
    return new self($userID);
    }

    public static function Create($data) {

    $id = Database::Insert(self::$databaseTable, $data);

    return new self($id);
    }

    public function Save() {
    $data = Utils::copyFromObject($this, $this->changedFields);

    Database::Update(self::$databaseTable, $data, $this->id);

    }

    public function __constructor() {
    // We do this to allow __get and __set to be called on public vars
    foreach(self::$databaseFields as $field) {
        unset($this->$field);
    }
    }

    public function __get($variableName) {
    return $this->$variableName;
    }

    public function __set($variableName, $variableValue) {
    // We only want to update what has been changed
    if(!in_array($variableName, $this->changedFields) && in_array($variableName, self::$databaseFields)) {
        array_push($this->changedFields, $variableName);
    }

    $this->$variableName = $variableValue;
    }
}

И вот код для одного из объектов, расширяющих базовый класс выше:

class Client extends DatabaseObject {

    public static $databaseTable = "clients";
    public static $databaseFields = array("name","contactName","primaryUserID","email","is_active","rg","cpf","cnpj","ie","addrType","addrName","addrNumber","addrComplement","addrPostalCode","addrNeighborhood","addrCity","addrState","addrCountry","phoneLandline","phoneFax","phoneMobile");

    public $name;
    public $contactName;

    public $primaryUserID;
    public $email;

    public $is_active;

    public $rg;
    public $cpf;
    public $cnpj;
    public $ie;

    public $addrType;
    public $addrName;
    public $addrNumber;
    public $addrComplement;
    public $addrPostalCode;
    public $addrNeighborhood;
    public $addrCity;
    public $addrState;
    public $addrCountry;

    public $phoneLandline;
    public $phoneFax;
    public $phoneMobile;

    public static function Load($id) {
 return new Client($id);
    }

}

Что я здесь делаю неправильно? Есть ли другой способ добиться такого же результата?

Краткое добавление: я объявляю атрибуты в классе класса главным образом, чтобы это можно было увидеть с помощью функции автозаполнения NetBeans.

Ответ 1

Вы ищете Поздняя статическая привязка.

Итак, вам нужно использовать:

static::$databaseTable

вместо

self::$databaseTable

Эта функция доступна с PHP 5.3. Симуляция этого в PHP 5.2 очень сложна из-за двух причин: get_called_class доступен только с PHP 5.3. Поэтому он также должен быть смоделирован с использованием debug_backtrace. Вторая проблема заключается в том, что если у вас есть вызываемый класс, вы все равно не можете использовать $calledClass::$property, потому что это тоже функция PHP 5.3. Здесь вам нужно использовать eval или Reflection. Поэтому я надеюсь, что у вас есть PHP 5.3;)