Когда левая часть является экземпляром объекта, вы используете ->
. В противном случае вы используете ::
.
Это означает, что ->
в основном используется для доступа к членам экземпляра (хотя он также может использоваться для доступа к статическим членам, такое использование не рекомендуется), в то время как::
обычно используется для доступа к статическим членам (хотя в некоторых особых случаях он используется для доступа к членам экземпляра). ).
Как правило, ::
используется для разрешения области видимости и может иметь либо имя класса parent
, self
либо, либо (в PHP 5.3) static
слева от него. parent
относится к области действия суперкласса класса, в котором он используется; self
относится к области действия класса, в котором он используется; static
относится к «вызываемой области видимости» (см. поздние статические привязки ).
Правило состоит в том, что вызов with ::
является вызовом экземпляра тогда и только тогда, когда:
- целевой метод не объявлен как статический и
- во время вызова существует совместимый контекст объекта, что означает, что они должны быть истинными:
- вызов сделан из контекста, где
$this
существует и
- класс
$this
- это либо класс вызываемого метода, либо его подкласс.
Пример:
class A {
public function func_instance() {
echo "in ", __METHOD__, "\n";
}
public function callDynamic() {
echo "in ", __METHOD__, "\n";
B::dyn();
}
}
class B extends A {
public static $prop_static = 'B::$prop_static value';
public $prop_instance = 'B::$prop_instance value';
public function func_instance() {
echo "in ", __METHOD__, "\n";
/* this is one exception where :: is required to access an
* instance member.
* The super implementation of func_instance is being
* accessed here */
parent::func_instance();
A::func_instance(); //same as the statement above
}
public static function func_static() {
echo "in ", __METHOD__, "\n";
}
public function __call($name, $arguments) {
echo "in dynamic $name (__call)", "\n";
}
public static function __callStatic($name, $arguments) {
echo "in dynamic $name (__callStatic)", "\n";
}
}
echo 'B::$prop_static: ', B::$prop_static, "\n";
echo 'B::func_static(): ', B::func_static(), "\n";
$a = new A;
$b = new B;
echo '$b->prop_instance: ', $b->prop_instance, "\n";
//not recommended (static method called as instance method):
echo '$b->func_static(): ', $b->func_static(), "\n";
echo '$b->func_instance():', "\n", $b->func_instance(), "\n";
/* This is more tricky
* in the first case, a static call is made because $this is an
* instance of A, so B::dyn() is a method of an incompatible class
*/
echo '$a->dyn():', "\n", $a->callDynamic(), "\n";
/* in this case, an instance call is made because $this is an
* instance of B (despite the fact we are in a method of A), so
* B::dyn() is a method of a compatible class (namely, it's the
* same class as the object's)
*/
echo '$b->dyn():', "\n", $b->callDynamic(), "\n";
Вывод:
B :: $ prop_static: B :: $ prop_static value
B :: func_static (): в B :: func_static
$ b-> prop_instance: B :: $ prop_instance value
$ b-> func_static (): in B :: func_static
$ B-> func_instance ():
в B :: func_instance
в A :: func_instance
в A :: func_instance
$ A-> динам ():
в A :: callDynamic
в динамическом dyn (__callStatic)
$ B-> динам ():
в A :: callDynamic
в динамическом dyn (__call)