由php的call_user_func传reference引发的思考


Posted in PHP onJuly 23, 2010

问题的提出
网友bercmisir在院内留言,针对php手册中的call_user_func函数的文档一事,大致如下:
http://php.net/manual/en/function.call-user-func.php

其中parameter下有这样一句话:
Note: Note that the parameters for call_user_func() are not passed by reference.

简单地翻译一下,是说这个函数的参数是不能依靠引用来传递的。

还有一个例子:

error_reporting(E_ALL); 
function increment(&$var) 
{ 
$var++; 
} 
$a = 0; 
call_user_func('increment', $a); 
echo $a."\n"; 
call_user_func_array('increment', array(&$a)); // You can use this instead before PHP 5.3 
echo $a."\n"; 
?>

输出是:
0
1

而网友bercmisir的问题在于:
call_user_func('increment', $a);输出是0,而call_user_func('increment', &$a);却输出是1,明明说不能依靠引用来传递。

寻根溯源
然后再进一步寻根溯源,这个Note的信息其实是http://bugs.php.net/bug.php?id=24931这个bug中最后处理的结果。
并且在call_user_func('increment', &$a);虽然输出了1的结果,但一般情况下,会有一个警告信息:Deprecated: Call-time pass-by-reference has been deprecated。

这是什么原因呢?
先看一个例子:

error_reporting(E_ALL); 
function increment(&$var) 
{ 
$var++; 
} 
$x = 1; 
increment($x); 
echo $x; 
?>

结果为2,并且没有类似expected to be a reference, value given的警告信息,相反地,如果将第8行代码修改为&$x,将得到一个废除警告。从而得以验证,其实PHP在传递过程中,变量会根据形参需要的到底是引用还是值来自行决定传输的是引用还是值,并不需要显式地传递(相反显式传递是即将被废除的)。

继续深入
http://www.php.net/manual/en/language.references.pass.php
在php手册中,介绍引用的传递一节,在中间位置有一个Note说到:在函数调用时是不需要传引用的(也就是上节所说的显式调用),在5.3中如果显式调用会出来一个废除警告。

分析源码
有人说:在php中写入,everything is a reference。
查阅php源码,在./Zend/zend_compile.c的1579行有函数定义zend_do_pass_param。(php5.2.13)

其中有这样一句判断:
if (original_op == ZEND_SEND_REF && !CG(allow_call_time_pass_reference)) {打印废除警告。}
大概意思就是说,在传递的是引用,并且php.ini的allow_call_time_pass_reference为否的话,打印警告。
再看zend_do_pass_param使用的地方,可以发现是在parser阶段时,根据参数ZVAL结构体中元素的定义,来传递到底是var还是value还是reference。(php5.2.13 ./Zend/zend_language_parser.y/c 451/3593)

结论
引用其实类似linux里的文件硬链接一样,但和C语言中的指针是不相同的,在parser阶段php会根据上下文环境自行判断是传引用还是值。而本文所提到的call_user_function并不会自行判断传的是引用还是值。所以前面的例子call_user_function在传值的时候不管用,而在传引用的时候得出了正确结果(但其实还有一个废除警告)。

PHP 相关文章推荐
透析PHP的配置文件php.ini
Oct 09 PHP
快速开发一个PHP扩展图文教程
Dec 12 PHP
PHP三元运算符的结合性介绍
Jan 10 PHP
PHP5中GD库生成图形验证码(有汉字)
Jul 28 PHP
PHP中的str_repeat函数在JavaScript中的实现
Sep 16 PHP
浅析PHP程序设计中的MVC编程思想
Jul 28 PHP
php获取发送给用户的header信息的方法
Mar 16 PHP
php通过array_shift()函数移除数组第一个元素的方法
Mar 18 PHP
PHP获取音频文件的相关信息
Jun 22 PHP
php读取qqwry.dat ip地址定位文件的类实例代码
Nov 15 PHP
简单实现php上传文件功能
Sep 21 PHP
PHP与Web页面交互操作实例分析
Jun 02 PHP
Google Voice 短信发送接口PHP开源版(2010.5更新)
Jul 22 #PHP
PHP 飞信好友免费短信API接口开源版
Jul 22 #PHP
PHP计划任务之关闭浏览器后仍然继续执行的函数
Jul 22 #PHP
PHP垃圾回收机制简单说明
Jul 22 #PHP
PHP多线程抓取网页实现代码
Jul 22 #PHP
php上传文件的增强函数
Jul 21 #PHP
php 模拟POST|GET操作实现代码
Jul 20 #PHP
You might like
php入门教程 精简版
2009/12/13 PHP
thinkPHP实现MemCache分布式缓存功能
2016/03/23 PHP
php简单实现短网址(短链)还原的方法(测试可用)
2016/05/09 PHP
Jquery选择器 $实现原理
2009/12/02 Javascript
Javascript学习笔记-详解in运算符
2011/09/13 Javascript
javascript suggest效果 自动完成实现代码分享
2012/02/17 Javascript
javascript学习笔记(二十) 获得和设置元素的特性(属性)
2012/06/20 Javascript
JavaScript面向对象知识串结(读JavaScript高级程序设计(第三版))
2012/07/17 Javascript
P3P Header解决Cookie跨域的问题
2013/03/12 Javascript
jquery实现文本框数量加减功能的例子分享
2014/05/10 Javascript
JS实现黑色大气的二级导航菜单效果
2015/09/18 Javascript
JavaScript第一篇之实现按钮全选、功能
2016/08/21 Javascript
javascript 实现动态侧边栏实例详解
2016/11/11 Javascript
MUI实现上拉加载和下拉刷新效果
2017/06/30 Javascript
Mobile Web开发基础之四--处理手机设备的横竖屏问题
2017/08/11 Javascript
浅谈Express异步进化史
2017/09/09 Javascript
Vue 2.0入门基础知识之内部指令详解
2017/10/15 Javascript
探讨Vue.js的组件和模板
2017/10/27 Javascript
React Native中NavigatorIOS组件的简单使用详解
2018/01/27 Javascript
Vue实现内部组件轮播切换效果的示例代码
2018/04/07 Javascript
Vue+Element实现表格编辑、删除、以及新增行的最优方法
2019/05/28 Javascript
通过实例解析vuejs如何实现调试代码
2020/07/16 Javascript
vscode 远程调试python的方法
2017/12/01 Python
Pytorch入门之mnist分类实例
2018/04/14 Python
Django框架使用内置方法实现登录功能详解
2019/06/12 Python
python实现滑雪游戏
2020/02/22 Python
django admin后管定制-显示字段的实例
2020/03/11 Python
解决Jupyter notebook中.py与.ipynb文件的import问题
2020/04/21 Python
python批量检查两个对应的txt文件的行数是否一致的实例代码
2020/10/31 Python
让IE9以下版本的浏览器兼容HTML5的方法
2014/03/12 HTML / CSS
Skyscanner波兰:廉价航班
2017/11/07 全球购物
中东奢侈品市场:Coveti
2019/05/12 全球购物
公司2015年终工作总结
2015/05/26 职场文书
商场广播稿范文
2015/08/19 职场文书
Python趣味挑战之用pygame实现简单的金币旋转效果
2021/05/31 Python
教你使用Ubuntu搭建DNS服务器
2022/09/23 Servers