Как переопределить статическое свойство родительского объекта и позволить родительскому объекту получить доступ к новому значению в PHP? - PullRequest
18 голосов
/ 22 июля 2010

Это то, что у меня есть: все объекты, которые можно сохранить в базе данных, расширяют абстрактный класс 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 Ответ

49 голосов
/ 22 июля 2010

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

Поэтому вам нужно использовать:

static::$databaseTable

вместо

self::$databaseTable

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

...