前言
首先,这个 static
符号跟 static(静态)关键字
不是一个东西。这三个符号在PHP对象中共有两种用法:
- 在类内部,可以使用
new self
、new static
、new parent
创建新对象 - 可以使用
self::
、static::
、parent::
调用静态变量和方法。
创建新对象
<?php class test{ ???public static function test_self(){ ???????return new self(); ???} ???public static function test_static(){ ???????return new static(); ???} ???public static function test_parent(){ ???????return new parent(); ???}}class test2 extends test{ ???public static function test_parent(){ ???????return new parent(); ???}}class test3 extends test2{}echo get_class(test::test_self()); ?//testecho get_class(test::test_static()); //testecho get_class(test2::test_self()); ?//testecho get_class(test2::test_static()); //test2echo get_class(test2::test_parent()); //testecho get_class(test3::test_self()); //testecho get_class(test3::test_static()); //test3 echo get_class(test3::test_parent()); //test
由以上这个例子可以得出:
new self
创建的对象是定义new self
的类创建的对象new static
创建的对象是执行new static
的类创建的对象new parent
创建的对象是定义new parent
的父类创建的对象(PHP5.3引进)
调用静态变量
概念
- 转发调用(forwarding call):所谓的"转发调用"指的是通过以下几种方式进行的静态调用:
self::
,parent::
,static::
以及forward_static_call()
.即在进行静态调用时未指名类名的调用属于转发调用。 - 非转发调度(non-forwarding call):非转发调用其实就是明确指定类名的静态调用(foo::bar())和非静态调用($foo->bar())。即明确地指定类名的静态调用和非静态调用。
- 后期静态绑定(Late Static Bindings ):"后期静态绑定"的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。
不存在继承的时候
self和static无区别。
- 在静态函数中,self和static可以调用静态属性和静态方法(沒有实例化类,因此不能呼叫非静态的属性和方法)。
在非静态函数中, self和static可以调用非静态属性和非静态方法。
<?php ?class Demo{ ??????????public static $static; ??????????public $Nostatic; ???????????public function __construct(){ ??????????????????self::$static = "static"; ??????????????????$this->Nostatic = "Nostatic"; ?????} ?????public static function get(){ ?????????return __CLASS__; ?????} ?????public function show(){ ?????????return "this is function show with ".$this->Nostatic; ?????} ????public function test(){ ????????echo Demo::$static."\n"; ?//使用类型调用静态属性 ????static ????????echo Demo::get()."\n"; ?//使用类名调用非静态方法 ???Demo ????????echo Demo::show()."\n"; ?//使用类名调用静态方法 ??this is function show with Nostatic ????????echo self::$static."\n"; ?//self调用静态属性 ??????static ????????echo self::get()."\n"; ?//self调用非静态方法 ?????Demo ????????echo self::show()."\n"; ??//self调用静态方法 ??????this is function show with Nostatic ????????echo static::$static."\n";//static调用静态属性 ????static ????????echo static::get()."\n";//static调用非静态方法 ???Demo ????????echo static::show()."\n"; //static调用静态方法 ??this is function show with Nostatic ?????} ?} $obj = new Demo(); $obj->test();
存在继承关系的时候
- self调用的方法和属性始终表示当前类的方法和属性
- static调用的方法和属性为当前执行的类的方法和属性
parent调用的方法和属性为父类的方法和属性
?<?php ???class A{ ????static $test = "AAA"; ????static function getClassName(){ ?????????return "A"; ?????} ?????function getClassName2(){ ?????????return "AA"; ?????} ?????static function testSelf(){ ?????????echo self::getClassName(); ?????????echo "\n"; ?????????echo self::getClassName2(); ???????????echo "\n"; ?????????echo self::$test; ?????} ????static function testStatic(){ ?????????echo static::getClassName(); ?????????echo "\n"; ?????????echo static::getClassName2(); ?????????echo "\n"; ?????????echo static::$test; ?????} ?} ?class B extends A{ ??????static $test = "BBB"; ??????static function getClassName(){ ???????????return "B"; ???????} ??????function getClassName2(){ ???????????return "BB"; ???????} ??????static function testParent(){ ???????????echo parent::getClassName(); ???????????echo "\n"; ???????????echo parent::getClassName2(); ???????????echo "\n"; ???????????echo parent::$test; ?????} ?} ??class C extends B{ ???????static $test = "CCC"; ???????static function getClassName(){ ???????????return "C"; ???????} ???????function getClassName2(){ ???????????return "CC"; ???????} ???????static function testParent(){ ????????????echo parent::getClassName(); ????????????echo "\n"; ????????????echo parent::getClassName2(); ????????????echo "\n"; ????????????echo parent::$test; ???????} ?} ?B::testSelf(); // A AA AAA ?echo "\n"; ?B::testStatic();// B BB BBB ?echo "\n"; ?B::testParent();// A AA AAA ?echo "\n"; ?C::testSelf(); // A AA AAA ?echo "\n"; ?C::testStatic(); // C CC CCC ?echo "\n"; ?C::testParent(); // B BB BBB
解析一下手册里的例子
<?php class A { ???????public static function foo() { ????????????????static::who(); ???????} ???????public static function who() { ???????????????echo __CLASS__."\n"; ???????} } class B extends A { ???????public static function test() { ???????????????A::foo(); ???????????????parent::foo(); ???????????????self::foo(); ???????} ???????public static function who() { ????????????????echo __CLASS__."\n"; ????????} } class C extends B { ????????public static function who() { ???????????????echo __CLASS__."\n"; ????????}} C::test(); ?>
最后输出为 A C C
单独拿出进行分析
public static function test() { ???????????A::foo(); ???????????parent::foo(); ???????????self::foo(); ????} ????
A::foo()
:非转发请求,直接调用A的foo()
方法,在任何地方调用结果都是一样的parent::foo()
:在B类中写着,调用B的父类A的方法foo()
(parent的用法);A类中的foo()
中执行static::who()
,寻找上一个非转发请求的类名(在A类的foo()
方法中写上get_called_class()
,可得为C,由此可知就是当前执行的类),所以调用C类的who()
方法(这一步就可以理解为后期静态绑定,即为static的用法);C类中重写了who()
方法,所以结果为C;如果去掉C类中的who()
方法,会调用B类中的who()
方法;如果再去掉B类中的who()
方法,会调用A类中的who()
方法。self::foo()
:执行B类中的foo()
方法(self的用法),B类中没有foo()
方法,于是继承了A类的foo()
方法,如果B类中定义了foo()
方法,则执行B类中的foo()
方法;执行A类的foo()
方法,如上.
PHP常见概念混淆(七)之self、static、parent的区别
原文地址:https://www.cnblogs.com/qiye5757/p/9437971.html