php5.3后静态绑定用法详解


Posted in PHP onNovember 11, 2016

本文实例讲述了php5.3后静态绑定用法。分享给大家供大家参考,具体如下:

手册原文:

自 PHP 5.3.0 起,PHP 增加了一个叫做后期静态绑定的功能,用于在继承范围内引用静态调用的类。

准确说,后期静态绑定工作原理是存储了在上一个"非转发调用"(non-forwarding call)的类名。当进行静态方法调用时,该类名即为明确指定的那个(通常在 :: 运算符左侧部分);当进行非静态方法调用时,即为该对象所属的类。所谓的"转发调用"(forwarding call)指的是通过以下几种方式进行的静态调用:self::,parent::,static:: 以及 forward_static_call()。可用 get_called_class() 函数来得到被调用的方法所在的类名,static:: 则指出了其范围

该功能从语言内部角度考虑被命名为"后期静态绑定"。"后期绑定"的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。也可以称之为"静态绑定",因为它可以用于(但不限于)静态方法的调用。

self:: 的限制

使用 self:: 或者 __CLASS__ 对当前类的静态引用,取决于定义当前方法所在的类:

Example #1 self:: 用法

<?php
class A {
public static function who() {
echo __CLASS__;
}
public static function test() {
self::who();
}
}
class B extends A {
public static function who() {
echo __CLASS__;
}
}
B::test();
?>

以上例程会输出:

A

后期静态绑定的用法 后期静态绑定本想通过引入一个新的关键字表示运行时最初调用的类来绕过限制。简单地说,这个关键字能够让你在上述例子中调用 test() 时引用的类是 B 而不是 A。最终决定不引入新的关键字,而是使用已经预留的 static 关键字。

Example #2 static:: 简单用法

<?php
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();
?>

以上例程会输出:

B

Note: 在非静态环境下,所调用的类即为该对象实例所属的类。由于 $this-> 会在同一作用范围内尝试调用私有方法,而 static:: 则可能给出不同结果。另一个区别是 static:: 只能用于静态属性。

Example #3 非静态环境下使用 static::

<?php
class A {
private function foo() {
echo "success!\n";
}
public function test() {
$this->foo();
static::foo();
}
}
class B extends A {
/* foo() will be copied to B, hence its scope will still be A and
* the call be successful */
}
class C extends A {
private function foo() {
/* original method is replaced; the scope of the new one is C */
}
}
$b = new B();
$b->test();
$c = new C();
$c->test(); //fails
?>

以上例程会输出:

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

Note: 后期静态绑定的解析会一直到取得一个完全解析了的静态调用为止。另一方面,如果静态调用使用 parent:: 或者 self:: 将转发调用信息。

Example #4 转发和非转发调用

<?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

下面示例分析了基于PHP后期静态绑定功能解决在继承范围内引用静态调用的类。

先看如下代码:

class Person
{
public static function status()
{
self::getStatus();
}
protected static function getStatus()
{
echo "Person is alive";
}
}
class Deceased extends Person
{
protected static function getStatus()
{
echo "Person is deceased";
}
}
Deceased::status(); //Person is alive

很明显,结果不是我们预期的,这是因为self::取决于定义时所在的类,而不是运行中的类。为了解决这个问题,你可能会在继承类中重写status()方法,更好的解决方案是PHP 5.3后添加了后期静态绑定的功能。

代码如下:

class Person
{
public static function status()
{
static::getStatus();
}
protected static function getStatus()
{
echo "Person is alive";
}
}
class Deceased extends Person
{
protected static function getStatus()
{
echo "Person is deceased";
}
}
Deceased::status(); //Person is deceased

可见,static::不在指向当前所在的类,实际上,它是在运行中计算的,强制获取最终类的所有属性。

因此,建议,以后不要再使用self::,使用static::

补充:

网友帖1

php的后期静态绑定,怎么解释?下面的这幅图输出是A,C,C

php5.3后静态绑定用法详解

由图的继承关系可知:C彻底包含了B和A。

在看答案结果以前,他细观察发现,三个类里都有同一个名称who()方法。
系统会用最后一个优先级最高,进一步的说,你几乎没法通过C去调用A、B内的who(),只能重改方法,比如添加个getBWho(){echo B::who();}
然后通过C::getBWho();来调用B内的who();

下面来看运行结果:

test只在B中出现,所以结果必然是test()中运行的三个结果:

第一个:静态直接指名到姓的调用A内静态函数,这没有悬念,必然是A
第二个:parent::是调用上一级的父类,在此题中为A,A中又直接调用static:who();上面说过了,这个who()优先级最高的在C里面,无论在你ABC中哪里调用,只要是static::who()必然是最后定义的那个,覆盖效应,如果想调用A里的必需指明A::who()或是通过去除static从作用域限制来实现。所以这个who()就是C中定义的who
第三个:self::who与第二个类似的问题,看样该走B的,注意覆盖效应,要想调用B内的who必须得B::who(),因为更高级的C已经重写了这个方法,如果C中没有who,肯定就是B,依次类推。所以必然还是调用C中的who;

所以答案为:ACC

代码如下:

<?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 B B

网友帖2

(还是针对上面图中的代码)

手册不是说得很清楚么

”后期绑定“的意思是说,static::不再被解析为定义当前方法所在的类,而是在实际运行时计算的。也可以称之为”静态绑定“,因为它可以用于(但不限于)静态方法的调用。

#1说的有个小问题

【self::foo(); // 这个self实际上是C类。明白吗? C::test() C继承了B的test()方法】

不准确,self还是B类,但是本身没有覆写foo方法,所以就调用父类A的foo方法。

如果self实际是C类,那你试下self::foo();改成self::who();,应当打印C,但是打印B,这也正是self和static的区别。

<?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::who();
}
public static function who() {
echo __CLASS__."\n";
}
}
class C extends B {
public static function who() {
echo __CLASS__."\n";
}
}
C::test();
?>

输出为:A C B

网友帖3

A::foo(); //A指代A类,访问A类的foo方法和who方法
parent::foo();//调用B类的父类——A的foo方法,并告诉foo方法最原始的调用者是C
self::foo(); //self指代定义该方法的类,即B,但是B没有定义foo方法,它将原始的调用者C向上传递,
// 访问父类的foo方法,最后访问c的who方法;

所以这就回答了楼上的疑问:若是把self::foo(); 改成self::who(),因为self指代B,而B有who方法,所以结果是变成了B

静态调用使用 parent:: 或者 self:: 将转发原始调用信息。

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
从Web查询数据库之PHP与MySQL篇
Sep 25 PHP
PHP has encountered an Access Violation 错误的解决方法
Jan 17 PHP
php shell超强免杀、减少体积工具实现代码
Oct 16 PHP
基于PHP array数组的教程详解
Jun 05 PHP
php输出金字塔的2种实现方法
Dec 16 PHP
ThinkPHP3.2.2的插件控制器功能
Mar 05 PHP
PHP中preg_match正则匹配中的/u、/i、/s含义
Apr 17 PHP
Centos PHP 扩展Xchche的安装教程
Jul 09 PHP
php fread读取文件注意事项
Sep 24 PHP
php ajax数据传输和响应方法
Aug 21 PHP
php装饰者模式简单应用案例分析
Oct 23 PHP
PHP中mysqli_get_server_version()的实例用法
Feb 03 PHP
php基于curl实现的股票信息查询类实例
Nov 11 #PHP
PHP中STDCLASS用法实例分析
Nov 11 #PHP
php遍历替换目录下文件指定内容的方法
Nov 10 #PHP
php实现有序数组打印或排序的方法【附Python、C及Go语言实现代码】
Nov 10 #PHP
PHP数组生成XML格式数据的封装类实例
Nov 10 #PHP
Linux平台php命令行程序处理管道数据的方法
Nov 10 #PHP
PHP中功能强大却很少使用的函数实例小结
Nov 10 #PHP
You might like
Windows Apache2.2.11及Php5.2.9-1的安装与配置方法
2009/06/08 PHP
遍历指定目录下的所有目录和文件的php代码
2011/11/27 PHP
微信公众平台接口开发入门示例
2014/12/24 PHP
PHP表单数据写入MySQL数据库的代码
2016/05/31 PHP
laravel框架实现为 Blade 模板引擎添加新文件扩展名操作示例
2020/01/25 PHP
jQuery 取值、赋值的基本方法整理
2014/03/31 Javascript
使用jQuery调用XML实现无刷新即时聊天
2016/08/07 Javascript
JS查找字符串中出现次数最多的字符
2016/09/05 Javascript
前端框架Vue.js中Directive知识详解
2016/09/12 Javascript
正则表达式替换html元素属性的方法
2016/11/26 Javascript
浅谈js中function的参数默认值
2017/02/20 Javascript
js实现城市级联菜单的2种方法
2017/06/23 Javascript
ES6模块化的import和export用法方法总结
2017/08/08 Javascript
安装vue-cli报错 -4058 的解决方法
2017/10/19 Javascript
浅析为什么a=&quot;abc&quot; 不等于 a=new String(&quot;abc&quot;)
2017/10/25 Javascript
js判断文件类型大小并给出提示的实现方法
2018/01/03 Javascript
利用Decorator如何控制Koa路由详解
2018/06/26 Javascript
JS div匀速移动动画与变速移动动画代码实例
2019/03/26 Javascript
微信小程序访问豆瓣电影api的实现方法
2019/03/31 Javascript
深度了解vue.js中hooks的相关知识
2019/06/14 Javascript
vue封装swiper代码实例解析
2019/10/08 Javascript
vue 解决form表单提交但不跳转页面的问题
2019/10/30 Javascript
[39:18]完美世界DOTA2联赛PWL S3 Forest vs LBZS 第二场 12.17
2020/12/19 DOTA
numpy concatenate数组拼接方法示例介绍
2019/05/27 Python
python paramiko远程服务器终端操作过程解析
2019/12/14 Python
python文件处理fileinput使用方法详解
2020/01/02 Python
python有序查找算法 二分法实例解析
2020/02/18 Python
Keras保存模型并载入模型继续训练的实现
2021/02/20 Python
CSS3 旋转立方体问题详解
2020/01/09 HTML / CSS
美国创意礼品网站:UncommonGoods
2017/02/03 全球购物
药学专业大学生自荐信
2013/09/28 职场文书
班长岗位职责
2013/11/10 职场文书
房地产营销策划方案
2014/02/08 职场文书
学生会感恩节活动方案
2014/10/11 职场文书
2015年化妆品销售工作总结
2015/05/11 职场文书
详解Spring Security如何在权限中使用通配符
2022/06/28 Java/Android