PHP的Calling Scope
作者: Laruence( ) 本文地址: http://www.laruence.com/2012/06/14/2628.html 转载请注明出处 昨天在Yaf交流群, 大草原同学批评我变懒了, Blog很久没更新了, 今天刚好有人在Segmentfalut上问了我一个问题, ?我在微博上也做了简单的解答, 不过感觉一句话说不
- 作者: Laruence(
)
- 本文地址: http://www.laruence.com/2012/06/14/2628.html
- 转载请注明出处
昨天在Yaf交流群, 大草原同学批评我变懒了, Blog很久没更新了, 今天刚好有人在Segmentfalut上问了我一个问题, ?我在微博上也做了简单的解答, 不过感觉一句话说不清楚, 就写篇blog凑个数吧.
问题在这里, ?因为太长, 我就不copy过来了: 这是php中__call和__callStatic在被继承后会产生的bug?
这个问题乍看, 确实很容易让人迷惑, 但实际上, 造成这样的误解的根本原因在于: 在PHP中, 判断静态与否不是靠”::”(PAAMAYIM_NEKUDOTAYIM)符号, 而是靠calling scope.
那么, 什么是calling scope?
在PHP中, 调用一个方法的时候, $this指针指向的对象就是这个方法被调用时刻的calling scope. 对于下面的例子:
<?php Foo::bar(); ?>
在调用bar方法的时候, 处于一个没有calling scope域的上下文中, 所以这个是静态调用.
而对于如下的例子:
<?php class A { public function test() { Foo::bar(); } } $a = new A(); $a->test();
在调用bar方法的时候, 处于一个$a对象的上下文中, 也就是说, 此时的calling scope是$a对象, 所以这个其实不是静态调用.
为了验证这一个结论, 请看下面的一个实际例子:
<?php class Foo { public function bar() { var_dump($this); } } class A { public function test() { Foo::bar(); } } $a = new A(); $a->test(); ?>
输出什么呢?
object(A)#1 (0) { }
在调用bar的时候, 这个看似”静态”调用的调用, $this指针却是被赋值的, 指向的是$a对象, 那么这个还算静态调用么?
我举这个例子是为了说明这个问题, 但大家在实际的应用中, 大家尽量要避免使用”::”来调用一个非静态的方法, PHP也会对于这种调用给出一个Strict 警告:
Strict Standards: Non-static method Foo::bar() should not be called statically, assuming $this from incompatible context
也许有人会说这个应该算bug吧? 其实不然, 更多的应该是错误使用造成的, 因为你在一个有calling scope的上下文中采用”静态的形式”调用了一个类的非静态方法所致.
那么PHP为什么要这么设计呢? 考虑下面的例子:
<?php class A { public function __construct() { } } class B extends A { public function __construct() { parent::__construct(); } }
当我们调用父类的构造函数的时候, 我们是有意的要把当前的scope传递给父类的构造函数作为calling scope的.
现在大家对静态调用, 是不是稍微能有更进一步的理解呢? 下午公司马上就要开全体大会, 匆忙而就, 写的可能有点乱, 请大家海涵, 呵呵, thanks
Comments
- 2012/06/14, 张洋 writes: 鸟哥V5,顶了再看...... :-P
- 2012/06/14, 张洋 writes: Demon: 13:33:28 更新blog的举动让一群活不下去的人重新站起来了! ......
- 2012/06/14, ujnjing writes: 鸟哥V5~
- 2012/06/14, 陆离 writes: 今天看到鸟哥的微博也觉得挺困惑的,正在纠结呢就有了这么一篇博文。学习了。
- 2012/06/14, jarfield writes: 个人觉得最后一个parent的例子说明PHP此特性的缘由,不是很有说服力。 Java/C++中,均有“子类构造函数调用父类构造函数”的需求。Java对super 关键字做了特殊处理。PHP也完全可以对parent::做特殊处理。类似于Foo::bar的误用,可以直接报错。 这样是否更加安全一些?毕竟很多PHP程序员均有Java/C++的背景,某些概念深入人心。
- 2012/06/14, xinqiyang writes: 鸟哥的yaf群号是啥? 主要是parent::__construct(); 和普通的 class::staticfunction 记住的了。。。。。
- 2012/06/14, 子痕 writes: Yaf 群号: 5134185,多谢鸟哥授业解惑,哈哈。
- 2012/06/14, shirne writes: 似乎明白了。 呵呵
- 2012/06/15, wizardmin writes: 哈哈,之前也认为是个BUG ”::”来调用一个非静态的方法还是应该给个warning错误
- 2012/06/15, bill writes: 现丑了。 最后一个例子个人觉得不能很好的说明calling scope。 class B 的实例化对像在调用构造函数的时候,同时也会调用父类(class B)的构造函数,parent::__construct(), 这里的parent不是当前B类的实例化对像,而是Class A本身。所以这里不会是把当前的calling scope传给父类。当前的calling scope是class B的实例化对像。如果传递给父类(class A),那不就是在class A里面用class B的实例化对像(calling scope)? 对于作用域,官方文档的OOP第章好像就有例子说明。 轻拍
- 2012/06/19, 陈辉云 writes: 报不报strict,看error_reporting的设置。
- 2012/06/19, huming17 writes: test(); ?> 静态调用函数 public function bar() 定义成 public static function bar() 再静态调用,$this指针不会被赋值。
- 2012/06/20, wclssdn writes: 给力~~~ 不过. 看到那个$this竟然有值.. 还是A 我就诧异了.... 如果$this->bar2(); 那岂不是报bar2不存在A中? 看来写法的规范性是必须的啊~~ 我就一直没碰到过这问题.. 哈哈哈~~~
- 2012/06/25, PHP爱好者 writes: 表示我是个规范的写代码的人,不过看了laruence的这篇文章,还是很有收获!
- 2012/06/25, funlake writes: 给力,又学到东西了,感谢。
- 2012/07/12, eason writes: 说实话,这种作用域的东西,在任何语言里都有。 一般对javascript 运行机制,作用域,作用域链有深刻理解的人,再来看php的作用域那就更容易了。
- 2012/08/09, 奇言妙事-文学奇谈小小说阅读xlinblog.sinaapp.com » Blog Archive » PHP的Calling Scope writes: [...] 本文地址: http://www.laruence.com/2012/06/14/2628.html [...]
- 2012/08/12, PHP的Calling Scope树林/咖啡 成都专业php网站制作 | 树林/咖啡 成都专业php网站制作 writes: [...] 风雪之隅 ? PHP应用 Posted in: php / Tagged: PHP的Calling, Scope [...]
- 2012/08/30, 疯狂的火星人 writes: 再看一遍!
- 2012/09/13, PHP静态调用非静态方法 | 编程·早晨 writes: [...] 好了,对于为什么会出现这个情况的实现原因, 鸟哥的这篇文章有说明 http://www.laruence.com/2012/06/14/2628.html [...]
- 2012/09/21, x6y6 writes: 用静态方法去调用一个非静态函数,这个为什么不干脆不报错呢,因为按照其他语言,比如Java、C++是直接报错的。
- 2012/09/21, x6y6 writes: 我有点奇怪,用调用静态方法的方法去调用一个非静态方法,为什么不干脆直接报错呢,因为按照其他语言,比如Java、C++这种情况是直接报错的。
- 2012/11/19, PHP静态调用非静态方法 « 成人免费资源三级分享网站 writes: [...] 好了,对于为什么会出现这个情况的实现原因, 鸟哥的这篇文章有说明 http://www.laruence.com/2012/06/14/2628.html [...]
- 2012/11/28, Jason writes: 学习了!!!
- 2013/03/15, FxYCarl writes: 怪不得我之前在用call_user_function调用对象的方法的时候会出现没在对象上下文的提示...
- 2013/05/03, noName writes: 我觉得这是在PHP语言设计上有问题。
Related posts:
- 如何获取一个变量的名字
- Javascript作用域原理
- 深入理解PHP原理之函数(Introspecting PHP Function)
- 深入理解PHP原理之变量分离/引用(Variables Separation)
- 深入浅出PHP(Exploring PHP)
Copyright © 2010 风雪之隅 版权所有, 转载务必注明. 该Feed只供个人使用, 禁止未注明的转载或商业应用. 非法应用的, 一切法律后果自负. 如有问题, 可发E-mail至my at laruence.com.(Digital Fingerprint: 73540ba0a1738d7d07d4b6038d5615e2)

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题











PHP 8.4 带来了多项新功能、安全性改进和性能改进,同时弃用和删除了大量功能。 本指南介绍了如何在 Ubuntu、Debian 或其衍生版本上安装 PHP 8.4 或升级到 PHP 8.4

如果您是一位经验丰富的 PHP 开发人员,您可能会感觉您已经在那里并且已经完成了。您已经开发了大量的应用程序,调试了数百万行代码,并调整了一堆脚本来实现操作

Visual Studio Code,也称为 VS Code,是一个免费的源代码编辑器 - 或集成开发环境 (IDE) - 可用于所有主要操作系统。 VS Code 拥有针对多种编程语言的大量扩展,可以轻松编写

JWT是一种基于JSON的开放标准,用于在各方之间安全地传输信息,主要用于身份验证和信息交换。1.JWT由Header、Payload和Signature三部分组成。2.JWT的工作原理包括生成JWT、验证JWT和解析Payload三个步骤。3.在PHP中使用JWT进行身份验证时,可以生成和验证JWT,并在高级用法中包含用户角色和权限信息。4.常见错误包括签名验证失败、令牌过期和Payload过大,调试技巧包括使用调试工具和日志记录。5.性能优化和最佳实践包括使用合适的签名算法、合理设置有效期、

本教程演示了如何使用PHP有效地处理XML文档。 XML(可扩展的标记语言)是一种用于人类可读性和机器解析的多功能文本标记语言。它通常用于数据存储

字符串是由字符组成的序列,包括字母、数字和符号。本教程将学习如何使用不同的方法在PHP中计算给定字符串中元音的数量。英语中的元音是a、e、i、o、u,它们可以是大写或小写。 什么是元音? 元音是代表特定语音的字母字符。英语中共有五个元音,包括大写和小写: a, e, i, o, u 示例 1 输入:字符串 = "Tutorialspoint" 输出:6 解释 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。总共有 6 个元

静态绑定(static::)在PHP中实现晚期静态绑定(LSB),允许在静态上下文中引用调用类而非定义类。1)解析过程在运行时进行,2)在继承关系中向上查找调用类,3)可能带来性能开销。

PHP的魔法方法有哪些?PHP的魔法方法包括:1.\_\_construct,用于初始化对象;2.\_\_destruct,用于清理资源;3.\_\_call,处理不存在的方法调用;4.\_\_get,实现动态属性访问;5.\_\_set,实现动态属性设置。这些方法在特定情况下自动调用,提升代码的灵活性和效率。
