由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 导出Excel示例分享
Aug 18 PHP
CentOS安装php v8js教程
Feb 26 PHP
PHP stream_context_create()函数的使用示例
May 12 PHP
PHP检测用户是否关闭浏览器的方法
Feb 14 PHP
Yii2框架RESTful API 格式化响应,授权认证和速率限制三部分详解
Nov 10 PHP
PHP通过引用传递参数用法分析
Dec 01 PHP
利用php + Laravel如何实现部署自动化详解
Oct 11 PHP
PHP基于DateTime类解决Unix时间戳与日期互转问题【针对1970年前及2038年后时间戳】
Jun 13 PHP
PHP基于DateTime类解决Unix时间戳与日期互转问题【针对1970年前及2038年后时间戳】
Jun 13 PHP
PHP SESSION机制的理解与实例
Mar 22 PHP
PHP常用函数之根据生日计算年龄功能示例
Oct 21 PHP
php实现断点续传大文件示例代码
Jun 19 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 switch语句多个值匹配同一代码块的实现
2014/03/03 PHP
在PHP模板引擎smarty生成随机数的方法和math函数详解
2014/04/24 PHP
java模拟PHP的pack和unpack类
2016/04/13 PHP
laravel中数据显示方法(默认值和下拉option默认选中)
2019/10/11 PHP
在JQuery dialog里的服务器控件 事件失效问题
2010/12/08 Javascript
JavaScript立即执行函数的三种不同写法
2014/09/05 Javascript
使用js获取图片原始尺寸
2014/12/03 Javascript
jQuery实现冻结表头的方法
2015/03/09 Javascript
javascript数组去重的方法汇总
2015/04/14 Javascript
JavaScript中Window对象的属性及事件
2015/12/25 Javascript
node.js实现博客小爬虫的实例代码
2016/10/08 Javascript
Vue.js第四天学习笔记
2016/12/02 Javascript
自学实现angularjs依赖注入
2016/12/20 Javascript
JSON键值对序列化和反序列化解析
2017/01/24 Javascript
vue-axios使用详解
2017/05/10 Javascript
Angular 4依赖注入学习教程之ValueProvider的使用(七)
2017/06/04 Javascript
限时抢购-倒计时的完整实例(分享)
2017/09/17 Javascript
JS实现不用中间变量temp 实现两个变量值得交换方法
2018/02/04 Javascript
nodejs更改项目端口号的方法
2018/05/13 NodeJs
微信小程序中时间戳和日期的相互转换问题
2018/07/09 Javascript
vue 更改连接后台的api示例
2019/11/11 Javascript
Vue 实现登录界面验证码功能
2020/01/03 Javascript
vue 解决setTimeOut和setInterval函数无效报错的问题
2020/07/30 Javascript
详细介绍Python的鸭子类型
2016/09/12 Python
python hbase读取数据发送kafka的方法
2018/12/27 Python
Python处理mysql特殊字符的问题
2020/03/02 Python
Python读写压缩文件的方法
2020/07/30 Python
英国高档百货连锁店:John Lewis
2017/11/20 全球购物
电子狗项圈:eDog Australia
2019/12/04 全球购物
Bata印度官网:源自欧洲舒适鞋履品牌
2020/01/30 全球购物
SQL Server里面什么样的视图才能创建索引
2015/04/17 面试题
个人自我评价和职业目标
2014/01/24 职场文书
名企HR怎样看待求职信
2014/02/23 职场文书
2015年信贷员工作总结
2015/04/28 职场文书
运动会开幕式主持词
2015/07/01 职场文书
JS实现刷新网页后之前浏览位置保持不变示例详解
2022/08/14 Javascript