亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何理解PHP函數的實現原理及性能

發布時間:2021-09-29 10:03:14 來源:億速云 閱讀:147 作者:iii 欄目:開發技術

本篇內容介紹了“如何理解PHP函數的實現原理及性能”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

php函數的分類

在php中,橫向劃分的話,函數分為兩大類: user function(內置函數) 和internal function(內置函數)。前者就是用戶在程序中自定義的一些函數和方法,后者則是php本身提供的各類庫函數(比如sprintf、array_push等)。用戶也可以通過擴展的方法來編寫庫函數,這個將在后面介紹。對于user function,又可以細分為function(函數)和method(類方法),本文中將就這三種函數分別進行分析和測試。

php函數的實現

一個php函數最終是如何執行,這個流程是怎么樣的呢?
要回答這個問題,我們先來看看php代碼的執行所經過的流程。

如何理解PHP函數的實現原理及性能

從圖1可以看到,php實現了一個典型的動態語言執行過程:拿到一段代碼后,經過詞法解析、語法解析等階段后,源程序會被翻譯成一個個指令(opcodes),然后ZEND虛擬機順次執行這些指令完成操作。Php本身是用c實現的,因此最終調用的也都是c的函數,實際上,我們可以把php看做是一個c開發的軟件。通過上面描述不難看出,php中函數的執行也是被翻譯成了opcodes來調用,每次函數調用實際上是執行了一條或多條指令。

對于每一個函數,zend都通過以下的數據結構來描述

復制代碼 代碼如下:


typedef union _zend_function {
zend_uchar type; /* MUST be the first element of this struct! */
struct {
zend_uchar type; /* never used */
char *function_name;
zend_class_entry *scope;
zend_uint fn_flags;
union _zend_function *prototype;
zend_uint num_args;
zend_uint required_num_args;
zend_arg_info *arg_info;
zend_bool pass_rest_by_reference;
unsigned char return_reference;
} common;

zend_op_array op_array;
zend_internal_function internal_function;
} zend_function;


typedef struct _zend_function_state {
HashTable *function_symbol_table;
zend_function *function;
void *reserved[ZEND_MAX_RESERVED_RESOURCES];
} zend_function_state;

其中type標明了函數的類型:用戶函數、內置函數、重載函數。Common中包含函數的基本信息,包括函數名,參數信息,函數標志(普通函數、靜態方法、抽象方法)等內容。另外,對于用戶函數,還有一個函數符號表,記錄了內部變量等,這個將在后面詳述。 Zend維護了一個全局function_table,這是一個大的hahs表。函數調用的時候會首先根據函數名從表中找到對應的zend_function。當進行函數調用時候,虛擬機會根據type的不同決定調用方法, 不同類型的函數,其執行原理是不相同的 。

內置函數

內置函數,其本質上就是真正的c函數,每一個內置函數,php在最終編譯后都會展開成為一個名叫zif_xxxx的function,比如我們常見的sprintf,對應到底層就是zif_sprintf。Zend在執行的時候,如果發現是內置函數,則只是簡單的做一個轉發操作。
Zend提供了一系列的api供調用,包括參數獲取、數組操作、內存分配等。內置函數的參數獲取,通過zend_parse_parameters方法來實現,對于數組、字符串等參數,zend實現的是淺拷貝,因此這個效率是很高的。可以這樣說,對于php內置函數,其效率和相應c函數幾乎相同,唯一多了一次轉發調用。

內置函數在php中都是通過so的方式進行動態加載,用戶也可以根據需要自己編寫相應的so,也就是我們常說的擴展。ZEND提供了一系列的api供擴展使用

用戶函數

和內置函數相比,用戶通過php實現的自定義函數具有完全不同的執行過程和實現原理。如前文所述,我們知道php代碼是被翻譯成為了一條條opcode來執行的,用戶函數也不例外,實際中每個函數對應到一組opcode,這組指令被保存在zend_function中。于是,用戶函數的調用最終就是對應到一組opcodes的執行。

》》局部變量的保存及遞歸的實現

我們知道,函數遞歸是通過堆棧來完成的。在php中,也是利用類似的方法來實現。Zend為每個php函數分配了一個活動符號表(active_sym_table),記錄當前函數中所有局部變量的狀態。所有的符號表通過堆棧的形式來維護,每當有函數調用的時候,分配一個新的符號表并入棧。當調用結束后當前符號表出棧。由此實現了狀態的保存和遞歸。
對于棧的維護,zend在這里做了優化。預先分配一個長度為N的靜態數組來模擬堆棧,這種通過靜態數組來模擬動態數據結構的手法在我們自己的程序中也經常有使用,這種方式避免了每次調用帶來的內存分配、銷毀。ZEND只是在函數調用結束時將當前棧頂的符號表數據clean掉即可。因為靜態數組長度為N,一旦函數調用層次超過N,程序不會出現棧溢出,這種情況下zend就會進行符號表的分配、銷毀,因此會導致性能下降很多。在zend里面,N目前取值是32。因此,我們編寫php程序的時候,函數調用層次最好不要超過32。當然,如果是web應用,本身可以函數調用層次的深度。

》》參數的傳遞 和內置函數調用zend_parse_params來獲取參數不同,用戶函數中參數的獲取是通過指令來完成的。函數有幾個參數就對應幾條指令。具體到實現上就是普通的變量賦值。通過上面的分析可以看出,和內置函數相比,由于是自己維護堆棧表,而且每條指令的執行也是一個c函數,用戶函數的性能相對會差很多,后面會有具體的對比分析。因此,如果一個功能有對應php內置函數實現的盡量不要自己重新寫函數去實現。

“如何理解PHP函數的實現原理及性能”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

php
AI

红安县| 大厂| 乡宁县| 竹北市| 海南省| 宁海县| 香河县| 宁乡县| 克东县| 桂阳县| 台东县| 靖江市| 红安县| 墨竹工卡县| 凌云县| 美姑县| 乌什县| 河间市| 武川县| 武汉市| 湛江市| 乐平市| 抚州市| 鄂州市| 郸城县| 北宁市| 乌拉特后旗| 汾阳市| 格尔木市| 黄浦区| 德安县| 武安市| 札达县| 大田县| 尚义县| 舞钢市| 花莲县| 芮城县| 安岳县| 甘德县| 平罗县|