PHP基礎終極版-面試大全
提示:本文章為系列文章,喜歡的請一鍵三連
本文:(一)深度理解變數的傳值和參考
提示:有任何問題請在評論區留言,謝謝
文章目錄
- PHP基礎終極版-面試大全
- 前言
- 一、PHP變數存盤?
- 變數容器
- 參考計數
- 計數步驟
- 傳遞變數給函式
- 變數傳遞給函式的步驟
- 二、PHP變數取值?
- 三、PHP變數賦值?
- 賦值和unset
- 兩種賦值的影響:
- 四、PHP變數普通傳值?
- 五、PHP變數參考傳值?
- 參考傳值舉例:
- 參考傳值決議:
- 六、PHP兩種傳值的區別?
- 七、PHP變數記憶體管理?
- 記憶體管理原則是:
- 決議:
- 總結
前言
本文基礎學習內容:
如何深度全面的理解PHP中的變數、及變數的傳值和參考知識點,讓你在作業中快速上手,成為技術強人,在面試中快速超越他人拿到offer,
提示:如果你正在面試,或在作業中遇到問題,可以來為自己加油,一起深度學下PHP的基礎,
一、PHP變數存盤?
變數容器
在PHP內部,變數是存盤在一個叫做zval的容器中,它不僅僅包含變數的值,也包含變數的型別,Python和PHP類似,也有一個標簽標記變數型別,變數容器中包含一些Zend引擎用來區分是否參考的欄位,同時它也包含這個值的參考計數,
變數存盤在一個相當于關聯陣列的符號表中,這個陣列以變數名為key,并且指向包含了這些變數的容器,
更簡單的說就是:變數名存盤在記憶體堆疊中,它是指向堆中具體記憶體的地址
參考計數
PHP試著在變數拷貝(如 $a = $b )的時候變得聰明些,“=”也稱為賦值運算子,當進行賦值操作時,Zend引擎不會創建一個新的變數視窗,而是增大變數視窗的 refcount 欄位,你可以想象一下,當這個變數是一個巨大的字串或一個巨大 的陣列時,這將節約多少的記憶體,
如下圖所示:
計數步驟
- 第一步: 變數a,包含文本”this is”,默認情況下,參考計數等于1
- 第二步:將變數$a賦值給$b和$c,這里沒有新的變數容器生成,僅僅是每次在變數賦值操作時將refcount加1,因為這里執行了兩次賦值操作,所以refcount最后會變成3,
現在,也許你很想知道當變數$c改變時將發生什么,根據refcount的值的不同,它會有兩種不同的處理方式,如果 refcount等于1,這個變數容器將更新它的值(也許同時會更新它的型別),如果refcount大于1,將創建一個包含了新值(和型別)的變數容器,如上圖所示的第三步,$a變數所在的變數容器的refcount值被減去1,現在refcount的值是2,而新創建的容器的refcount的值為 1,當對一個變數使用unset函式時,這個變數所在的容器的refcount值將減去1,如上圖第4步所示,如果refcount的值少于1,Zend引 擎將翻譯這個變數容器,如圖第5步所示,
傳遞變數給函式
除了所有腳本共用的全域符號表以外,每個用戶定義的函式在呼叫時都會創建一個屬于自己的符號表,用來存放它自己的變數,當一個函式被呼叫后,Zend引擎 就會創建一個這樣的符號表,當這個函式回傳時這個函式表就會被釋放,一個函式要么通過return陳述句回傳,要么因為函式結束而回傳(譯者注:無回傳的函 數默認會回傳NULL),
如下圖所示:

也許你對上圖有看不懂,沒關系,下面來詳細說下,
下面通過上圖我們來詳細介紹變數是如何傳遞給函式的,
變數傳遞給函式的步驟
- 第一步,我們將”thisis”賦給變數$a,然后我們將這個變數傳遞do_something()函式的$s變數,
- 第二步,你可以看到這與變數賦值的操作是一樣的(與我們在前一小節提到的$b = $a類似),只是其存盤在不同的符號表(函式符號表),并且參考計數加2,而不是加1,原因是函式堆疊也包含了這個變數容器的參考,
- 第三步,當我們賦新值給變數$s,原變數容器的refcount減1,并且創建一個包含了新值的變數容器,
- 第四步,我們通過return陳述句回傳一個變數,回傳的變數從全域符號表中獲取一個物體并將其refcount的值增加1.當函式結束時,函式的符 號表將被銷毀,在銷毀的程序中,Zend引擎將遍歷符號表中的每個變數,并將其refcount的值減少,當變數容器的refount的值變為0,這個變數容器將會被銷毀,
如你所見,由于 PHP的參考計數機制,變數容器不是以拷貝的方式從函式回傳,如果變數$s在第三步時沒有被修改,則變數$a和$b將一直指向相同的變數容器(這個容器的 refcount為2),在這種情況下,陳述句$a = “this is”將不會創建變數容器的副本,
二、PHP變數取值?
通過上面的了解,我們可以知道變數取值時是從變數容器中去取的,
也就是通過變數名查找堆中的記憶體
三、PHP變數賦值?
賦值和unset
$a = 1;
unset($a);
兩種賦值的影響:
參考傳值后unset被賦值的變數舉例:
$a = 1;
$b = &$a;
unset($a);
dump($a); // null
dump($b); //int 1
四、PHP變數普通傳值?
普通傳值,傳值以后,是不同的地址名稱,指向不同的記憶體物體$a = 1;
dump($a); //int 1
$a=2;
dump($a); //int 2
五、PHP變數參考傳值?
參考傳值,傳參考后,是不同的地址名稱,但都指向同一個記憶體物體;改變其中一個,另外一個就也被改變參考傳值舉例:
$a = 1;
$b = &$a;
dump($b); //int 1
$a=2;
dump($b); //int 2
參考傳值決議:
$b=&$a表示:$b找到$a指向的記憶體物體1,并建立$b與1的指標關系,
此時,如果$a改變指向,$b就會自動追隨$a改變指向,
注意:變數名的指向是記憶體地址,不是另一個變數,
六、PHP兩種傳值的區別?
普通傳值,傳值以后,是不同的地址名稱,指向不同的記憶體物體; 參考傳值,傳參考后,是不同的地址名稱,但都指向同一個記憶體物體;改變其中一個,另外一個就也被改變;七、PHP變數記憶體管理?
PHP的記憶體釋放是由計數器決定的記憶體管理原則是:
- 1, unset只是變數計數器減1,,當計數器為0的時候會觸發回收,
- 2, $s=null 會直接觸發回收,
決議:
在傳參考的正常情況下,如果\$a改變指向,\$b就會自動追隨\$a改變指向,但是如果\$a與記憶體的指標關系被注銷,\$b不需要做出任何變化,仍然指向原記憶體地址,unset($a)是delete掉$a和記憶體物體1的指標關系,由于記憶體物體1還在被另一個變數$b參考著,所以記憶體物體1不會被回收,所以$a沒有了指向的記憶體地址,值就是null,$b仍然指向原記憶體地址,
總結
以上就是今天要講的內容,本文僅僅簡單介紹了PHP變數的傳值和參考,而PHP的基礎知識點非常多,大家可以一起跟隨本系列文章學習更多更全面的PHP基礎,
轉載請註明出處,本文鏈接:https://www.uj5u.com/ruanti/404178.html
標籤:其他
上一篇:git 連接gitee時報錯 Auth error: Access deined: authorize failure
下一篇:電腦初始化


