?
Ce document utilise Manuel du site Web PHP chinois Libérer
全局變量
PHP中在函數(shù)、類之外直接定義的變量可以在函數(shù)、類成員方法中通過(guò)global關(guān)鍵詞引入使用,這些變量稱為:全局變量。
這些直接在PHP中定義的變量(包括include、require文件中的)相對(duì)于函數(shù)、類方法而言它們是全局變量,但是對(duì)自身執(zhí)行域zend_execute_data而言它們是普通的局部變量,自身執(zhí)行時(shí)它們與普通變量的讀寫(xiě)方式完全相同。
function test() { global $id; $id++; } $id = 1; test(); echo $id;
全局變量初始化
全局變量在整個(gè)請(qǐng)求執(zhí)行期間始終存在,它們保存在EG(symbol_table)中,也就是全局變量符號(hào)表,與靜態(tài)變量的存儲(chǔ)一樣,這也是一個(gè)哈希表,主腳本(或include、require)在zend_execute_ex執(zhí)行開(kāi)始之前會(huì)把當(dāng)前作用域下的所有局部變量添加到EG(symbol_table)中,這一步操作后面介紹zend執(zhí)行過(guò)程時(shí)還會(huì)講到,這里先簡(jiǎn)單提下:
ZEND_API void zend_execute(zend_op_array *op_array, zval *return_value) { ... i_init_execute_data(execute_data, op_array, return_value); zend_execute_ex(execute_data); ... } i_init_execute_data()這個(gè)函數(shù)中會(huì)把局部變量插入到EG(symbol_table): ZEND_API void zend_attach_symbol_table(zend_execute_data *execute_data) { zend_op_array *op_array = &execute_data->func->op_array; HashTable *ht = execute_data->symbol_table; if (!EXPECTED(op_array->last_var)) { return; } zend_string **str = op_array->vars; zend_string **end = str + op_array->last_var; //局部變量數(shù)組起始位置 zval *var = EX_VAR_NUM(0); do{ zval *zv = zend_hash_find(ht, *str); //插入全局變量符號(hào)表 zv = zend_hash_add_new(ht, *str, var); //哈希表中value指向局部變量的zval ZVAL_INDIRECT(zv, var); ... }while(str != end); }
從上面的過(guò)程可以很直觀的看到,在執(zhí)行前遍歷局部變量,然后插入EG(symbol_table),EG(symbol_table)中的value直接指向局部變量的zval,示例經(jīng)過(guò)這一步的處理之后(此時(shí)局部變量只是分配了zval,但還未初始化,所以是IS_UNDEF):
與靜態(tài)變量的訪問(wèn)一樣,全局變量也是將原來(lái)的值轉(zhuǎn)換為引用,然后在global導(dǎo)入的作用域內(nèi)創(chuàng)建一個(gè)局部變量指向該引用:
global $id; // 相當(dāng)于:$id = & EG(symbol_table)["id"];
具體的操作過(guò)程不再細(xì)講,與靜態(tài)變量的處理過(guò)程一致,這時(shí)示例中局部變量與全局變量的引用情況如下圖。
超全局變量
全部變量除了通過(guò)global引入外還有一類特殊的類型,它們不需要使用global引入而可以直接使用,這些全局變量稱為:超全局變量。
超全局變量實(shí)際是PHP內(nèi)核定義的一些全局變量:$GLOBALS、$_SERVER、$_REQUEST、$_POST、$_GET、$_FILES、$_ENV、$_COOKIE、$_SESSION、argv、argc。
銷毀
局部變量如果沒(méi)有手動(dòng)銷毀,那么在函數(shù)執(zhí)行結(jié)束時(shí)會(huì)將它們銷毀,而全局變量則是在整個(gè)請(qǐng)求結(jié)束時(shí)才會(huì)銷毀,即使是我們直接在PHP腳本中定義在函數(shù)外的那些變量。
void shutdown_destructors(void){ if (CG(unclean_shutdown)) { EG(symbol_table).pDestructor = zend_unclean_zval_ptr_dtor; } zend_try { uint32_t symbols; do { symbols = zend_hash_num_elements(&EG(symbol_table)); //銷毀 zend_hash_reverse_apply(&EG(symbol_table), (apply_func_t) zval_call_destructor); } while (symbols != zend_hash_num_elements(&EG(symbol_table))); } ... }