摘要:然而實際上,這個結果反而是正確的。結論我認為出現這種詭異的結果應該算是的,同時,這也說明了此種問題不太容易被發現和暴露,這要求我們平常寫代碼的時候盡量使用常用的語法,和精干的語句,讓代碼和邏輯達到最佳的平衡點。
前言
之前我在面試的時候,遇到許多年輕人都聲稱自己精通php,有過許多項目經驗等等。然而,當真正筆試的時候,我問到
$result=1; if(-1){ $result=2; } echo $result;
中,$result最終結果的時候,許多人信誓旦旦的告訴我是1。 試想,這樣一個連基本算法都搞不清楚的人,即使有過再多的項目經驗,你敢用嗎?
引申對于算法的一些問題,我個人一向是非常較真的,我招人的時候也是非常側重此方面,因此我對php的關注也是在這方面多些。故事得從一個知乎上的問題開始。
//第一題
//第二題
這個問題非常的有意思,也是一個大坑,許多人都算錯了,包括很多我認識的大牛。在這里就不點名字,以免其羞愧。 然而你以為我一開始算對了嗎?我算對了第一題,第二題卻是百思不得其解。最后用調試工具一番調試才算理清頭緒。索性發出來與大家一起分享這個有意思的問題。
分析 第一題 php語言解釋這個其實非常簡單,++a這種單目運算符的運算結果還是自身。 所以
$a=1; $b=&$a; echo (++$a)+(++$a); //換種寫法就等同于 $a=1; $a=++$a; //2 $a=++$a; //3 $a=$a+$a;//3+3=6
哈,很多人肯定以為是等于5,然而這個是操作的同一個變量,等同于改變了兩次$a的值,最后相加的時候,自然就是改變后的值相加,所以等于6。
正常情況然而在php中,為了照顧人類的邏輯,默認情況下,即使名字相同的基本類型的變量,也不會使用同一個變量地址,因此,以上代碼會被解析為
$a=1; echo (++$a)+(++$a); //換種寫法就等同于 $a=1; $a=++$a; //2 $b=++$a; //3 $a=$a+$b;//2+3=5
但是由于
$b=&$a;
的存在,使得下面的第一個$a的計算方式變成了傳統的c語言計算方式,所以輸出了_看起來錯誤的結果_。
然而實際上,這個結果反而是正確的。PHP中的糖語法寵壞了那些基礎本來不扎實的孩子,對這種加法做了特別的運算處理而已。
為什么說等于6才是正確結果呢?我們知道現行高級語言大多來自c語言,php也不例外,我們這里用c語言來寫一遍上述代碼,然后通過反匯編來看看機器到底是怎么執行的。
其實無論是否注釋下面的取地址,結果都是6。 我們看匯編代碼
這里更清晰的看到a的值的變化。 即是
mov edx,dword ptr [a] ;a的值為3 add edx,dword ptr [a] ; 3+3問題二 問題發現
由問題一的結論來分析問題二,反而陷入了一個更大的舞曲,為什么呢?
echo (++$a)+(++$a)+(++$a); //10 /* 按照問題一的分析,此處的結果應為 a=1+1 //2 a=a+1 //3 a=a+1 //4 a=a+a+a //12 */
然而輸出的結果卻是10.
動手實驗這里我們用一個工具phpdebug.exe來調試下看看。
可以看到的a的值的變化:
1->2->3->4
然后我們再來調試一下注釋掉
$b=&$a;
的結果。
這里在第二步的時候重新給了變量$a一個地址,實際上同是叫$a,其實他們已經不是同一個變量了。 所以他輸出的結果為9。
實驗結果但是為什么上面的結果為10呢? 這其實是因為
$b=&$a;
這個取地址運算只起效了一句運算指令,就是只管事了第一回合,對于以后的運算,php還是用了平常的算法。 即:
$a=1; $b=&$a; echo (++$a)+(++$a)+(++$a); /* 這段實際上是 $a=++$a; //2 $a=++$a; //3 //注意了,前兩個已經得到結果了,第三個我們用一個新的變量$c。 $c=++$a; //4 $a=$a+$a; //3+3=6 $a=$a+$c; //6+4=10 */結論
我認為出現這種詭異的結果應該算是php的bug,同時,這也說明了此種問題不太容易被發現和暴露,這要求我們平常寫代碼的時候盡量使用常用的語法,和精干的語句,讓代碼和邏輯達到最佳的平衡點。 此BUG我已經反饋到php官方。
后續最新的php7中已經修復了此bug。
修訂記錄初稿 2015-09-11
修訂 2016-05-27
修訂 2016-05-30
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/21741.html
摘要:且聽下回分解基于動態規劃策略的優化解法參見程序員的算法趣題偷懶的算盤上一篇程序員的算法趣題同數包夾程序員的算法趣題同數包夾本系列總目錄參見程序員的算法趣題詳細分析和全解程序員的算法趣題詳細分析和全解 目錄 1. 問題描述 2. 解題分析 3. 代碼及測試 4. 后記 1. 問題描述 ??...
摘要:過去一年時間寫了多篇文章來探討了我認為的框架最核心部分的設計思路代碼實現。為了大家閱讀方便,我把這些源碼學習的文章匯總到這里。數據庫算法和數據結構這些都是編程的內功,只有內功深厚了才能解決遇到的復雜問題。 過去一年時間寫了20多篇文章來探討了我認為的Larave框架最核心部分的設計思路、代碼實現。通過更新文章自己在軟件設計、文字表達方面都有所提高,在剛開始決定寫Laravel源碼分析地...
摘要:轉載自我的博客在眾多的新特性中,我覺得這是最神奇甚至是詭異的一個,如果有不理解這個概念的朋友,可能連它的說明都看不懂。在這個例子中操作符被重載為,操作符被重載為。再重載成操作符,以后咋跟蹤代碼。。。 轉載自我的博客:http://70.io/2014/03/php-5_6-internal-operator-overloading 在眾多php 5.6的新特性中,我覺得這是最...
摘要:最近面試他人的過程中,問了一些關于引用的知識,發現很多同學對這方面知之甚少,還有很多工作中基本沒有使用過。沒錢給大家發紅包,給大家推薦一家上海的好公司。對于上海的小伙伴或者想去上海的小伙伴,強烈建議去看看。 真的是變懶了,一個月一篇的節湊都很難保證了。 最近面試他人的過程中,問了一些關于PHP引用的知識,發現很多同學對這方面知之甚少,還有很多工作中基本沒有使用過。甚至有人告訴我要少用引...
閱讀 3367·2021-11-04 16:10
閱讀 3871·2021-09-29 09:43
閱讀 2706·2021-09-24 10:24
閱讀 3362·2021-09-01 10:46
閱讀 2514·2019-08-30 15:54
閱讀 594·2019-08-30 13:19
閱讀 3241·2019-08-29 17:19
閱讀 1062·2019-08-29 16:40