由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 相关文章推荐
关于Intype一些小问题的解决办法
Mar 28 PHP
PHP的基本常识小结
Jul 05 PHP
windows下PHP_intl.dll正确配置方法(apache2.2+php5.3.5)
Jan 14 PHP
yii框架表单模型使用及以数组形式提交表单数据示例
Apr 30 PHP
destoon供应信息title调用出公司名称的方法
Aug 22 PHP
PHP实现简单爬虫的方法
Jul 29 PHP
详解WordPress中用于合成数组的wp_parse_args()函数
Dec 18 PHP
基于PHP实现商品成交时发送短信功能
May 11 PHP
php blowfish加密解密算法
Jul 02 PHP
PHP通过微信跳转的Code参数获取用户的openid(关键代码)
Jul 06 PHP
Yii2使用表单上传文件的实例代码
Aug 03 PHP
PHP手机短信验证码实现流程详解
May 17 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实现求解最长公共子串问题的方法
2017/11/17 PHP
从ThinkPHP3.2.3过渡到ThinkPHP5.0学习笔记图文详解
2019/04/03 PHP
jquery ajax学习笔记2 使用XMLHttpRequest对象的responseXML
2011/10/16 Javascript
JQuery实现简单时尚快捷的气泡提示插件
2012/12/20 Javascript
js验证是否为数字的总结
2013/04/14 Javascript
js和jquery对dom节点的操作(创建/追加)
2013/04/21 Javascript
js写的方法实现上传图片之后查看大图
2014/03/05 Javascript
zepto中使用swipe.js制作轮播图附swipeUp,swipeDown不起效果问题
2015/08/27 Javascript
简单讲解jQuery中的子元素过滤选择器
2016/04/18 Javascript
onmouseover事件和onmouseout事件全面理解
2016/08/15 Javascript
深入解析Vue 组件命名那些事
2017/07/18 Javascript
ligerUI---ListBox(列表框可移动的实例)
2017/11/28 Javascript
详解vue使用vue-layer-mobile组件实现toast,loading效果
2018/08/31 Javascript
vue+element创建动态的form表单及动态生成表格的行和列
2019/05/20 Javascript
vue动态绘制四分之三圆环图效果
2019/09/03 Javascript
详解小程序云开发攻略(解决最棘手的问题)
2019/09/30 Javascript
JavaScript判断数据类型有几种方法及区别介绍
2020/09/02 Javascript
[08:42]DOTA2每周TOP10 精彩击杀集锦vol.2
2014/06/25 DOTA
python实现自动登录人人网并访问最近来访者实例
2014/09/26 Python
python实现中文分词FMM算法实例
2015/07/10 Python
Python自动发邮件脚本
2017/03/31 Python
Python基于回溯法子集树模板解决旅行商问题(TSP)实例
2017/09/05 Python
matplotlib subplots 调整子图间矩的实例
2018/05/25 Python
python http基本验证方法
2018/12/26 Python
解决Django中调用keras的模型出现的问题
2019/08/07 Python
python多线程分块读取文件
2019/08/29 Python
浅谈Django2.0 加xadmin踩的坑
2019/11/15 Python
Python3.7黑帽编程之病毒篇(基础篇)
2020/02/04 Python
《雨点》教学反思
2014/02/12 职场文书
初中家长寄语
2014/04/02 职场文书
感恩教师节演讲稿
2014/09/03 职场文书
擅自离岗检讨书
2014/09/12 职场文书
2014年局领导班子自身建设情况汇报
2014/11/21 职场文书
2014年电厂工作总结
2014/12/04 职场文书
2016幼儿园教师年度考核评语
2015/12/01 职场文书
使用RedisTemplat实现简单的分布式锁
2021/11/20 Redis