Quando a parte esquerda é uma instância de objeto, você usa ->
. Caso contrário, você usa ::
.
Isso significa que ->
é usado principalmente para acessar membros da instância (embora também possa ser usado para acessar membros estáticos, esse uso é desencorajado), enquanto ::
geralmente é usado para acessar membros estáticos (embora, em alguns casos especiais, seja usado para acessar membros da instância )
Em geral, ::
é usado para a resolução do âmbito , e que podem ter um nome de classe, parent
, self
, ou (em PHP 5.3) static
para o seu lado esquerdo. parent
refere-se ao escopo da superclasse da classe em que é usada; self
refere-se ao escopo da classe em que é usada; static
refere-se ao "chamado escopo" (consulte ligações estáticas tardias ).
A regra é que uma chamada com ::
é uma chamada de instância se e somente se:
- o método de destino não é declarado como estático e
- existe um contexto de objeto compatível no momento da chamada, o que significa que eles devem ser verdadeiros:
- a chamada é feita a partir de um contexto em que
$this
existe e
- a classe de
$this
é a classe do método que está sendo chamado ou uma subclasse dele.
Exemplo:
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";
Resultado:
B :: $ prop_static: B :: $ prop_static value
B :: func_static (): em B :: func_static
$ b-> prop_instance: B :: $ prop_instance value
$ b-> func_static (): em B :: func_static
$ b-> func_instance ():
em B :: func_instance
em A :: func_instance
em A :: func_instance
$ a-> dyn ():
em A :: callDynamic
no dyn dinâmico (__callStatic)
$ b-> dyn ():
em A :: callDynamic
no dyn dinâmico (__call)