一区二区久久-一区二区三区www-一区二区三区久久-一区二区三区久久精品-麻豆国产一区二区在线观看-麻豆国产视频

PHP錯誤抑制符(@)導致引用傳參失敗Bug的分析

看下面的例子:
復制代碼 代碼如下:
<?php
$array = array(1,2,3);
function add (&$arr) {
$arr[] = 4;
}
add(@$array);
print_r($array);
/**
此時, $array沒有改變, 輸出:
Array
(
[0] => 1
[1] => 2
[2] => 3
)
*/
add($array);
print_r($array);
/**
不使用錯誤抑制的情況下, 輸出正常:
Array
(
[0] => 1
[1] => 2
[2] => 3
[3] => 4
)
*/
?>

這個問題, 我之前沒有遇到過, 所以首先去找找相關資料, 看看有沒有現成的答案, Goolge了一番, 發現雖然有人已經向php報了類似的Bug:http://bugs.php.NET/bug.php?id=47623, 但php官方還沒有解決, 也沒有給出答復.

沒辦法, 只能自己分析了, 之前我曾經在文章中介紹過錯誤抑制符的原理( 深入理解php原理之錯誤抑制與內嵌HTML), 從原理上來說, 錯誤抑制只是修改了error_reporting的level, 按理來說不會影響到上下文之間的函數調用的機制. 只能通過實地試驗了.

經過gdb跟蹤, 發現在使用了錯誤移植符以后, 函數調用前的傳參opcode不同:
復制代碼 代碼如下:
//沒有使用錯誤抑制符的時候
OPCODE = SEND_REF
//使用了錯誤抑制符號以后
OPCODE = SEND_VAR_NO_RE

問題初步定位了, 但是造成這種差異的原因又是什么呢?

既然OPCODE不同, 那么肯定是在語法分析的階段, 走了不同的分支了, 想到這一層, 問題也就好定位了,

原來, php語法分析階段, 把形如 “@”+expr的條目, 規約成了expr_without_variable, 而這種節點的意義就是沒有變量的值, 也就是字面值, 我們都知道字面值是不能傳遞引用的(因為它不是變量), 所以, 就會導致這種差異.

具體過程如下:
1. 語法分析階段:
復制代碼 代碼如下:
expr_without_variable:
//...有省略
| '@' { zend_do_begin_silence(&$1 TSRMLS_CC); }
expr { zend_do_end_silence(&$1 TSRMLS_CC); $$ = $3; }
//此處走了ZEND_SEND_VAL分支
non_empty_function_call_parameter_list:
expr_without_variable { ....} //錯誤的走了這個分支
| variable {..... } //正常情況

所以導致在編譯期間, 生成了不同的OPCODE, 也導致了問題的表象.
最后, 我已經把原因在php的這個bug頁做了說明, 有興趣的可以去看看我的爛英語水平. 最后謝謝cici網友提供的這個有趣的問題.

php技術PHP錯誤抑制符(@)導致引用傳參失敗Bug的分析,轉載需保留來源!

鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。

主站蜘蛛池模板: 91精品推荐 | 国产亚洲漂亮白嫩美女在线 | 亚洲伦理一二三四 | 亚洲综合激情六月婷婷在线观看 | 国产乱来视频 | 思思玖玖玖在线精品视频 | 77777网站| 日韩午夜在线观看 | 欧美激情在线免费 | 国产成人短视频 | 中文字幕一级毛片 | 激情图片视频小说 | 色老板在线视频一区二区 | 精品极品三级久久久久 | 国产高清精品久久久久久久 | 欧美变态老妇重口与另类 | 精品久久久99大香线蕉 | 一级a性色生活片久久毛片 一级爱爱片一级毛片-一毛 | 日本wwwww视频 | 婷婷亚洲综合五月天小说在线 | 久久精品国产99久久99久久久 | 黄网站在线免费 | 亚洲天堂久久 | 91欧美激情一区二区三区成人 | 中文字幕一区二区三区四区五区人 | 国产女人成人精品视频 | 尤物网址在线观看日本 | 国产精品欧美在线不卡 | 国产大片免费观看中文字幕 | 秋霞伊人网 | 国产精品免费观看视频 | 97人人模人人爽人人喊6 | 在线激情网址 | 国内精品中文字幕 | 亚洲第一区精品日韩在线播放 | 欧美另类xxxx图片 | 国产一区二区三区在线观看精品 | 日韩视频第一页 | 青草国产在线 | 久久精品免费全国观看国产 | 黄色网址在线视频 |