Статические свойства и методы в PHP. Позднее статическое связывание Php статические методы вместо функций

Очень важной особенностью ООП является наличие статических свойств и методов . Главное, что необходимо сразу понять, что такие свойства и методы принадлежат не объекту, а классу . Это нужно понять с самого начала, а вот применение статических свойств и методов в PHP я рассмотрю в этой статье.

Самый классический пример - это класс, отвечающий за математические функции. Если кто-то знает Java , то он знает, что там имеется класс Math JavaScript такой класс тоже есть), содержащий множество математических функций. И там методы являются статическими. Это означает, что для того, чтобы посчитать какой-нибудь синус или экспоненту, не нужно создавать объект данного класса, что очень удобно.

Давайте мы с Вами напишем уменьшенную копию данного класса, но только для PHP:

class Math {
private static $count = 0;
public function __construct() {
self::$count++;
}
public static function calcSin($x) {
return sin($x);
}
public static function calcSQRT($x) {
return sqrt($x);
}
public static function getCount() {
return self::$count;
}
}
echo Math::calcSin(1);
echo "
";
echo Math::calcSQRT(9);
echo "
";
$math = new Math();
$math_2 = new Math();
echo Math::getCount();
?>

В данном коде я показал использование статических методов и свойств . Обратите внимание, что я реализовал классический способ счётчика объектов. Это получилось лишь благодаря тому, что поле count является статическим, и оно имеет одно и то же значение для всех объектов.

Ещё один популярный пример использования статических методов и свойств - это ведение лога. Все записи добавляются через статические методы. Также очень часто делают класс, состоящий из множества настроек, и там также все поля являются статическими. Как видите, примеров использования статических методов и свойств в PHP и других языках более, чем достаточно, поэтому уметь с ними работать надо обязательно.

Mar 23, 2010 dec5e

В PHP 5.3 появилась такая интересная возможность, как позднее статическое связывание (late static binding). Дальше немного вольный перевод описания из официального мануала .

Начиная с PHP 5.3.0 в языке реализована возможность, называемая поздним статическим связыванием, которая может использоваться для ссылки на вызываемый класс в контексте статического наследования.

Эту возможность назвали «позднее статическое связывание». «Позднее связывание» говорит о том, что static:: будет разрешаться не относительно класса, где определен метод, но будет вычисляться во время выполнения. «Статическое связывание» означает, что оно может быть использовано в вызовах статических методов (но не ограничивается только ими).

Ограничения self::

Пример № 1: использование self::

Пример выведет:

Использование позднего статического связывания

Позднее статическое связывание пытается решить это ограничение, вводя ключевое слово, ссылающееся на класс, первоначально вызванный в процессе выполнения. То есть, ключевое слово, которое позволит сослаться на B из test() в предыдущем примере. Было решено не вводить новое слово, а использовать уже зарезервированное static .

Пример № 2: простое использование static::

Пример выведет:

Замечание: static:: не работает как $this для статических методов! $this-> следует правилам наследования, а static:: нет. Это различие уточняется ниже.

Пример № 3: использование static:: в нестатическом контексте

test(); ?>

Пример выведет:

Замечание: Позднее статическое связывание останавливает процесс разрешения вызова. Статические вызовы с использованием ключевых слов parent:: или self:: передают информацию о вызове дальше.

Пример № 4: Передача и непередача вызовов

Пример выведет

Крайние случаи

Существует множество различных способов вызвать метод в PHP, такие как коллбэки или магически методы. Поскольку позднее статическое связывание разрешается во время выполнения, это может привести к неожиданным результатам в так называемых крайних случаях.

Пример № 5 Позднее статическое связывание в магических методах

foo; ?>

  • Программирование ,
  • ООП
    • Tutorial

    Не секрет, что на собеседованиях любят задавать каверзные вопросы. Не всегда адекватные, не всегда имеющие отношение к реальности, но факт остается фактом - задают. Конечно, вопрос вопросу рознь, и иногда вопрос, на первый взгляд кажущийся вам дурацким, на самом деле направлен на проверку того, насколько хорошо вы знаете язык, на котором пишете.

    Попробуем разобрать «по косточкам» один из таких вопросов - что значит слово «static» в PHP и зачем оно применяется?

    Ключевое слово static имеет в PHP три различных значения. Разберем их в хронологическом порядке, как они появлялись в языке.

    Значение первое - статическая локальная переменная

    function foo() { $a = 0; echo $a; $a = $a + 1; } foo(); // 0 foo(); // 0 foo(); // 0

    В PHP переменные локальны. Это значит, что переменная, определенная и получившая значение внутри функции (метода), существует только во время выполнения этой функции (метода). При выходе из метода локальная переменная уничтожается, а при повторном входе - создается заново. В коде выше такой локальной переменной является переменная $a - она существует только внутри функции foo() и каждый раз при вызове этой функции создается заново. Инкремент переменной в этом коде бессмысленен, поскольку на следующей же строчке кода функция закончит свою работу и значение переменной будет потеряно. Сколько бы раз мы не вызвали функцию foo(), она всегда будет выводить 0…

    Однако всё меняется, если мы перед присваиванием поставим ключевое слово static:

    Function foo() { static $a = 0; echo $a; $a = $a + 1; } foo(); // 0 foo(); // 1 foo(); // 2

    Ключевое слово static, написанное перед присваиванием значения локальной переменной, приводит к следующим эффектам:

    1. Присваивание выполняется только один раз, при первом вызове функции
    2. Значение помеченной таким образом переменной сохраняется после окончания работы функции
    3. При последующих вызовах функции вместо присваивания переменная получает сохраненное ранее значение
    Такое использование слова static называется статическая локальная переменная .
    Подводные камни статических переменных
    Разумеется, как всегда в PHP, не обходится без «подводных камней».

    Камень первый - статической переменной присваивать можно только константы или константные выражения. Вот такой код:
    static $a = bar();
    с неизбежностью приведет к ошибке парсера. К счастью, начиная с версии 5.6 стало допустимым присвоение не только констант, но и константных выражений (например - «1+2» или ""), то есть таких выражений, которые не зависят от другого кода и могут быть вычислены на этапе компиляции

    Камень второй - методы существуют в единственном экземпляре.
    Тут всё чуть сложнее. Для понимания сути приведу код:
    class A { public function foo() { static $x = 0; echo ++$x; } } $a1 = new A; $a2 = new A; $a1->foo(); // 1 $a2->foo(); // 2 $a1->foo(); // 3 $a2->foo(); // 4
    Вопреки интуитивному ожиданию «разные объекты - разные методы» мы наглядно видим на этом примере, что динамические методы в PHP «не размножаются». Даже если у нас будет сто объектов этого класса, метод будет существовать лишь в одном экземпляре, просто при каждом вызове в него будет пробрасываться разный $this.

    Такое поведение может быть неожиданным для неподготовленного к нему разработчика и послужить источником ошибок. Нужно заметить, что наследование класса (и метода) приводит к тому, что всё-таки создается новый метод:

    Class A { public function foo() { static $x = 0; echo ++$x; } } class B extends A { } $a1 = new A; $b1 = new B; $a1->foo(); // 1 $b1->foo(); // 1 $a1->foo(); // 2 $b1->foo(); // 2

    Вывод: динамические методы в PHP существуют в контексте классов, а не объектов. И только лишь в рантайме происходит подстановка "$this = текущий_объект"

    Значение второе - статические свойства и методы классов

    В объектной модели PHP существует возможность задавать свойства и методы не только для объектов - экземпляров класса, но и для класса в целом. Для этого тоже служит ключевое слово static:

    Class A { public static $x = "foo"; public static function test() { return 42; } } echo A::$x; // "foo" echo A::test(); // 42
    Для доступа к таким свойствам и методам используются конструкции с двойным двоеточием («Paamayim Nekudotayim»), такие как ИМЯ_КЛАССА::$имяПеременной и ИМЯ_КЛАССА:: имяМетода().

    Само собой разумеется, что у статических свойств и статических методов есть свои особенности и свои «подводные камни», которые нужно знать.

    Особенность первая, банальная - нет $this. Собственно это проистекает из самого определения статического метода - поскольку он связан с классом, а не объектом, в нём недоступна псевдопеременная $this, указывающая в динамических методах на текущий объект. Что совершенно логично.

    Однако, нужно знать, что в отличие от других языков, PHP не определяет ситуацию «в статическом методе написано $this» на этапе парсинга или компиляции. Подобная ошибка может возникнуть только в рантайме, если вы попытаетесь выполнить код с $this внутри статического метода.

    Код типа такого:
    class A { public $id = 42; static public function foo() { echo $this->id; } }
    не приведет ни к каким ошибкам, до тех пор, пока вы не попытаетесь использовать метод foo() неподобающим образом:
    $a = new A; $a->foo(); (и сразу получите «Fatal error: Using $this when not in object context»)

    Особенность вторая - static не аксиома!
    class A { static public function foo() { echo 42; } } $a = new A; $a->foo();
    Вот так, да. Статический метод, если он не содержит в коде $this, вполне можно вызывать в динамическом контексте, как метод объекта. Это не является ошибкой в PHP.

    Обратное не совсем верно:
    class A { public function foo() { echo 42; } } A::foo();
    Динамический метод, не использующий $this, можно выполнять в статическом контексте. Однако вы получите предупреждение «Non-static method A::foo() should not be called statically» уровня E_STRICT. Тут решать вам - или строго следовать стандартам кода, или подавлять предупреждения. Первое, разумеется, предпочтительнее.

    И кстати, всё написанное выше относится только к методам. Использование статического свойства через "->" невозможно и ведет к фатальной ошибке.

    Значение третье, кажущееся самым сложным - позднее статическое связывание

    Разработчики языка PHP не остановились на двух значениях ключевого слова «static» и в версии 5.3 добавили еще одну «фичу» языка, которая реализована тем же самым словом! Она называется «позднее статическое связывание» или LSB (Late Static Binding).

    Понять суть LSB проще всего на несложных примерах:

    Class Model { public static $table = "table"; public static function getTable() { return self::$table; } } echo Model::getTable(); // "table"
    Ключевое слово self в PHP всегда значит «имя класса, где это слово написано». В данном случае self заменяется на класс Model, а self::$table - на Model::$table.
    Такая языковая возможность называется «ранним статическим связыванием». Почему ранним? Потому что связывание self и конкретного имени класса происходит не в рантайме, а на более ранних этапах - парсинга и компиляции кода. Ну а «статическое» - потому что речь идет о статических свойствах и методах.

    Немного изменим наш код:

    Class Model { public static $table = "table"; public static function getTable() { return self::$table; } } class User extends Model { public static $table = "users"; } echo User::getTable(); // "table"

    Теперь вы понимаете, почему PHP ведёт себя в этой ситуации неинтуитивно. self был связан с классом Model тогда, когда о классе User еще ничего не было известно, поэтому и указывает на Model.

    Как быть?

    Для решения этой дилеммы был придуман механизм связывания «позднего», на этапе рантайма. Работает он очень просто - достаточно вместо слова «self» написать «static» и связь будет установлена с тем классом, который вызывает данный код, а не с тем, где он написан:
    class Model { public static $table = "table"; public static function getTable() { return static::$table; } } class User extends Model { public static $table = "users"; } echo User::getTable(); // "users"

    Это и есть загадочное «позднее статическое связывание».

    Нужно отметить, что для большего удобства в PHP кроме слова «static» есть еще специальная функция get_called_class(), которая сообщит вам - в контексте какого класса в данный момент работает ваш код.

    Удачных собеседований!

    • Tutorial

    Не секрет, что на собеседованиях любят задавать каверзные вопросы. Не всегда адекватные, не всегда имеющие отношение к реальности, но факт остается фактом - задают. Конечно, вопрос вопросу рознь, и иногда вопрос, на первый взгляд кажущийся вам дурацким, на самом деле направлен на проверку того, насколько хорошо вы знаете язык, на котором пишете.

    Попробуем разобрать «по косточкам» один из таких вопросов - что значит слово «static» в PHP и зачем оно применяется?

    Ключевое слово static имеет в PHP три различных значения. Разберем их в хронологическом порядке, как они появлялись в языке.

    Значение первое - статическая локальная переменная

    function foo() { $a = 0; echo $a; $a = $a + 1; } foo(); // 0 foo(); // 0 foo(); // 0

    В PHP переменные локальны. Это значит, что переменная, определенная и получившая значение внутри функции (метода), существует только во время выполнения этой функции (метода). При выходе из метода локальная переменная уничтожается, а при повторном входе - создается заново. В коде выше такой локальной переменной является переменная $a - она существует только внутри функции foo() и каждый раз при вызове этой функции создается заново. Инкремент переменной в этом коде бессмысленен, поскольку на следующей же строчке кода функция закончит свою работу и значение переменной будет потеряно. Сколько бы раз мы не вызвали функцию foo(), она всегда будет выводить 0…

    Однако всё меняется, если мы перед присваиванием поставим ключевое слово static:

    Function foo() { static $a = 0; echo $a; $a = $a + 1; } foo(); // 0 foo(); // 1 foo(); // 2

    Ключевое слово static, написанное перед присваиванием значения локальной переменной, приводит к следующим эффектам:

    1. Присваивание выполняется только один раз, при первом вызове функции
    2. Значение помеченной таким образом переменной сохраняется после окончания работы функции
    3. При последующих вызовах функции вместо присваивания переменная получает сохраненное ранее значение
    Такое использование слова static называется статическая локальная переменная .
    Подводные камни статических переменных
    Разумеется, как всегда в PHP, не обходится без «подводных камней».

    Камень первый - статической переменной присваивать можно только константы или константные выражения. Вот такой код:
    static $a = bar();
    с неизбежностью приведет к ошибке парсера. К счастью, начиная с версии 5.6 стало допустимым присвоение не только констант, но и константных выражений (например - «1+2» или ""), то есть таких выражений, которые не зависят от другого кода и могут быть вычислены на этапе компиляции

    Камень второй - методы существуют в единственном экземпляре.
    Тут всё чуть сложнее. Для понимания сути приведу код:
    class A { public function foo() { static $x = 0; echo ++$x; } } $a1 = new A; $a2 = new A; $a1->foo(); // 1 $a2->foo(); // 2 $a1->foo(); // 3 $a2->foo(); // 4
    Вопреки интуитивному ожиданию «разные объекты - разные методы» мы наглядно видим на этом примере, что динамические методы в PHP «не размножаются». Даже если у нас будет сто объектов этого класса, метод будет существовать лишь в одном экземпляре, просто при каждом вызове в него будет пробрасываться разный $this.

    Такое поведение может быть неожиданным для неподготовленного к нему разработчика и послужить источником ошибок. Нужно заметить, что наследование класса (и метода) приводит к тому, что всё-таки создается новый метод:

    Class A { public function foo() { static $x = 0; echo ++$x; } } class B extends A { } $a1 = new A; $b1 = new B; $a1->foo(); // 1 $b1->foo(); // 1 $a1->foo(); // 2 $b1->foo(); // 2

    Вывод: динамические методы в PHP существуют в контексте классов, а не объектов. И только лишь в рантайме происходит подстановка "$this = текущий_объект"

    Значение второе - статические свойства и методы классов

    В объектной модели PHP существует возможность задавать свойства и методы не только для объектов - экземпляров класса, но и для класса в целом. Для этого тоже служит ключевое слово static:

    Class A { public static $x = "foo"; public static function test() { return 42; } } echo A::$x; // "foo" echo A::test(); // 42
    Для доступа к таким свойствам и методам используются конструкции с двойным двоеточием («Paamayim Nekudotayim»), такие как ИМЯ_КЛАССА::$имяПеременной и ИМЯ_КЛАССА:: имяМетода().

    Само собой разумеется, что у статических свойств и статических методов есть свои особенности и свои «подводные камни», которые нужно знать.

    Особенность первая, банальная - нет $this. Собственно это проистекает из самого определения статического метода - поскольку он связан с классом, а не объектом, в нём недоступна псевдопеременная $this, указывающая в динамических методах на текущий объект. Что совершенно логично.

    Однако, нужно знать, что в отличие от других языков, PHP не определяет ситуацию «в статическом методе написано $this» на этапе парсинга или компиляции. Подобная ошибка может возникнуть только в рантайме, если вы попытаетесь выполнить код с $this внутри статического метода.

    Код типа такого:
    class A { public $id = 42; static public function foo() { echo $this->id; } }
    не приведет ни к каким ошибкам, до тех пор, пока вы не попытаетесь использовать метод foo() неподобающим образом:
    $a = new A; $a->foo(); (и сразу получите «Fatal error: Using $this when not in object context»)

    Особенность вторая - static не аксиома!
    class A { static public function foo() { echo 42; } } $a = new A; $a->foo();
    Вот так, да. Статический метод, если он не содержит в коде $this, вполне можно вызывать в динамическом контексте, как метод объекта. Это не является ошибкой в PHP.

    Обратное не совсем верно:
    class A { public function foo() { echo 42; } } A::foo();
    Динамический метод, не использующий $this, можно выполнять в статическом контексте. Однако вы получите предупреждение «Non-static method A::foo() should not be called statically» уровня E_STRICT. Тут решать вам - или строго следовать стандартам кода, или подавлять предупреждения. Первое, разумеется, предпочтительнее.

    И кстати, всё написанное выше относится только к методам. Использование статического свойства через "->" невозможно и ведет к фатальной ошибке.

    Значение третье, кажущееся самым сложным - позднее статическое связывание

    Разработчики языка PHP не остановились на двух значениях ключевого слова «static» и в версии 5.3 добавили еще одну «фичу» языка, которая реализована тем же самым словом! Она называется «позднее статическое связывание» или LSB (Late Static Binding).

    Понять суть LSB проще всего на несложных примерах:

    Class Model { public static $table = "table"; public static function getTable() { return self::$table; } } echo Model::getTable(); // "table"
    Ключевое слово self в PHP всегда значит «имя класса, где это слово написано». В данном случае self заменяется на класс Model, а self::$table - на Model::$table.
    Такая языковая возможность называется «ранним статическим связыванием». Почему ранним? Потому что связывание self и конкретного имени класса происходит не в рантайме, а на более ранних этапах - парсинга и компиляции кода. Ну а «статическое» - потому что речь идет о статических свойствах и методах.

    Немного изменим наш код:

    Class Model { public static $table = "table"; public static function getTable() { return self::$table; } } class User extends Model { public static $table = "users"; } echo User::getTable(); // "table"

    Теперь вы понимаете, почему PHP ведёт себя в этой ситуации неинтуитивно. self был связан с классом Model тогда, когда о классе User еще ничего не было известно, поэтому и указывает на Model.

    Как быть?

    Для решения этой дилеммы был придуман механизм связывания «позднего», на этапе рантайма. Работает он очень просто - достаточно вместо слова «self» написать «static» и связь будет установлена с тем классом, который вызывает данный код, а не с тем, где он написан:
    class Model { public static $table = "table"; public static function getTable() { return static::$table; } } class User extends Model { public static $table = "users"; } echo User::getTable(); // "users"

    Это и есть загадочное «позднее статическое связывание».

    Нужно отметить, что для большего удобства в PHP кроме слова «static» есть еще специальная функция get_called_class(), которая сообщит вам - в контексте какого класса в данный момент работает ваш код.

    Удачных собеседований!

    Начиная с версии PHP 5.3.0 появилась особенность, называемая позднее статическое связывание, которая может быть использована для того чтобы получить ссылку на вызываемый класс в контексте статического наследования.

    Если говорить более точно, позднее статическое связывание сохраняет имя класса указанного в последнем "не перенаправленном вызове". В случае статических вызовов это явно указанный класс (обычно слева от оператора :: ); в случае не статических вызовов это класс объекта. "Перенаправленный вызов" - это статический вызов, начинающийся с self:: , parent:: , static:: , или, если двигаться вверх по иерархии классов, forward_static_call() . Функция get_called_class() может быть использована чтобы получить строку с именем вызванного класса, и static:: представляет ее область действия.

    Само название "позднее статическое связывание" отражает в себе внутреннюю реализацию этой особенности. "Позднее связывание" отражает тот факт, что обращения через static:: не будут вычисляться по отношению к классу, в котором вызываемый метод определен, а будут вычисляться на основе информации в ходе исполнения. Также эта особенность была названа "статическое связывание" потому, что она может быть использована (но не обязательно) в статических методах.

    Ограничения self::

    Пример #1 Использование self::

    class A {
    echo __CLASS__ ;
    }
    public static function
    test () {
    self :: who ();
    }
    }

    class B extends A {
    public static function who () {
    echo __CLASS__ ;
    }
    }

    B :: test ();
    ?>

    Использование позднего статического связывания

    Позднее статическое связывание пытается устранить это ограничение предоставляя ключевое слово, которое ссылается на класс, вызванный непосредственно в ходе выполнения. Попросту говоря, ключевое слово, которое позволит вам ссылаться на B из test() в предыдущем примере. Было решено не вводить новое ключевое слово, а использовать static , которое уже зарезервировано.

    Пример #2 Простое использованиеstatic::

    class A {
    public static function who () {
    echo __CLASS__ ;
    }
    public static function
    test () {
    static:: who (); // Здесь действует позднее статическое связывание
    }
    }

    class B extends A {
    public static function who () {
    echo __CLASS__ ;
    }
    }

    B :: test ();
    ?>

    Результат выполнения данного примера:

    Замечание :

    В нестатическом контексте вызванным классом будет тот, к которому относится экземпляр объекта. Поскольку $this-> будет пытаться вызывать закрытые методы из той же области действия, использование static:: может дать разные результаты. Другое отличие в том, что static:: может ссылаться только на статические поля класса.

    Пример #3 Использование static:: в нестатическом контексте

    class A {
    private function foo () {
    echo "success!\n" ;
    }
    public function test () {
    $this -> foo ();
    static:: foo ();
    }
    }

    class B extends A {
    /* foo() будет скопирован в В, следовательно его область действия по прежнему А,
    и вызов будет успешен*/
    }

    class C extends A {
    private function foo () {
    /* исходный метод заменен; область действия нового метода С */
    }
    }

    $b = new B ();
    $b -> test ();
    $c = new C ();
    $c -> test (); //не верно
    ?>

    Результат выполнения данного примера:

    success! success! success! Fatal error: Call to private method C::foo() from context "A" in /tmp/test.php on line 9

    Замечание :

    Разрешающая область позднего статического связывания будет фиксирована вычисляющем ее статическим вызовом. С другой стороны, статические вызовы с использованием таких директив как parent:: или self:: перенаправляют информацию вызова.

    Пример #4 Перенаправленные и не перенаправленные вызовы

    Последние материалы раздела:

    Почему режется скорость Интернета по WiFi: Бесплатные советы как ускорить передачу данных
    Почему режется скорость Интернета по WiFi: Бесплатные советы как ускорить передачу данных

    Плохая скорость интернета через роутер - одна из наиболее «популярных» проблем всех любителей беспроводного соединения . В предыдущих статьях мы...

    Контекстное меню в Windows
    Контекстное меню в Windows

    Из этой информационной статьи вы узнаете о том, как вызвать контекстное меню для любого файла, папки, ярлыка и т.п используя для этого несколько...

    Продвижение в Instagram: самая подробная инструкция
    Продвижение в Instagram: самая подробная инструкция

    XXI век - стремительно меняющийся и ломающий прежние представления об успехе. Социальные сети стали феноменом, люди часами проводят время в режиме...