## 2.4 全局變量
PHP中在函數、類之外直接定義的變量可以在函數、類成員方法中通過global關鍵詞引入使用,這些變量稱為:全局變量。
這些直接在PHP中定義的變量(包括include、require文件中的)相對于函數、類方法而言它們是全局變量,但是對自身執行域zend_execute_data而言它們是普通的局部變量,自身執行時它們與普通變量的讀寫方式完全相同。
```php
function test() {
global $id;
$id++;
}
$id = 1;
test();
echo $id;
```
### 2.4.1 全局變量初始化
全局變量在整個請求執行期間始終存在,它們保存在`EG(symbol_table)`中,也就是全局變量符號表,與靜態變量的存儲一樣,這也是一個哈希表,主腳本(或include、require)在`zend_execute_ex`執行開始之前會把當前作用域下的所有局部變量添加到`EG(symbol_table)`中,這一步操作后面介紹zend執行過程時還會講到,這里先簡單提下:
```c
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()`這個函數中會把局部變量插入到EG(symbol_table):
```c
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;
//局部變量數組起始位置
zval *var = EX_VAR_NUM(0);
do{
zval *zv = zend_hash_find(ht, *str);
//插入全局變量符號表
zv = zend_hash_add_new(ht, *str, var);
//哈希表中value指向局部變量的zval
ZVAL_INDIRECT(zv, var);
...
}while(str != end);
}
```
從上面的過程可以很直觀的看到,在執行前遍歷局部變量,然后插入EG(symbol_table),EG(symbol_table)中的value直接指向局部變量的zval,示例經過這一步的處理之后(此時局部變量只是分配了zval,但還未初始化,所以是IS_UNDEF):

### 2.4.2 全局變量的訪問
與靜態變量的訪問一樣,全局變量也是將原來的值轉換為引用,然后在global導入的作用域內創建一個局部變量指向該引用:
```php
global $id; // 相當于:$id = & EG(symbol_table)["id"];
```
具體的操作過程不再細講,與靜態變量的處理過程一致,這時示例中局部變量與全局變量的引用情況如下圖。

### 2.4.3 超全局變量
全部變量除了通過global引入外還有一類特殊的類型,它們不需要使用global引入而可以直接使用,這些全局變量稱為:超全局變量。
超全局變量實際是PHP內核定義的一些全局變量:$GLOBALS、$_SERVER、$_REQUEST、$_POST、$_GET、$_FILES、$_ENV、$_COOKIE、$_SESSION、argv、argc。
### 2.4.4 銷毀
局部變量如果沒有手動銷毀,那么在函數執行結束時會將它們銷毀,而全局變量則是在整個請求結束時才會銷毀,即使是我們直接在PHP腳本中定義在函數外的那些變量。
```c
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)));
}
...
}
```
- 前言
- 第1章 PHP基本架構
- 1.1 PHP簡介
- 1.2 PHP7的改進
- 1.3 FPM
- 1.3.1 概述
- 1.3.2 基本實現
- 1.3.3 FPM的初始化
- 1.3.4 請求處理
- 1.3.5 進程管理
- 1.4 PHP執行的幾個階段
- 第2章 變量
- 2.1 變量的內部實現
- 2.2 數組
- 2.3 靜態變量
- 2.4 全局變量
- 2.5 常量
- 第3章 Zend虛擬機
- 3.1 PHP代碼的編譯
- 3.1.1 詞法解析、語法解析
- 3.1.2 抽象語法樹編譯流程
- 3.2 函數實現
- 3.2.1 內部函數
- 3.2.2 用戶函數的實現
- 3.3 Zend引擎執行流程
- 3.3.1 基本結構
- 3.3.2 執行流程
- 3.3.3 函數的執行流程
- 3.3.4 全局execute_data和opline
- 3.4 面向對象實現
- 3.4.1 類
- 3.4.2 對象
- 3.4.3 繼承
- 3.4.4 動態屬性
- 3.4.5 魔術方法
- 3.4.6 類的自動加載
- 3.5 運行時緩存
- 3.6 Opcache
- 3.6.1 opcode緩存
- 3.6.2 opcode優化
- 3.6.3 JIT
- 第4章 PHP基礎語法實現
- 4.1 類型轉換
- 4.2 選擇結構
- 4.3 循環結構
- 4.4 中斷及跳轉
- 4.5 include/require
- 4.6 異常處理
- 第5章 內存管理
- 5.1 Zend內存池
- 5.2 垃圾回收
- 第6章 線程安全
- 6.1 什么是線程安全
- 6.2 線程安全資源管理器
- 第7章 擴展開發
- 7.1 概述
- 7.2 擴展的實現原理
- 7.3 擴展的構成及編譯
- 7.3.1 擴展的構成
- 7.3.2 編譯工具
- 7.3.3 編寫擴展的基本步驟
- 7.3.4 config.m4
- 7.4 鉤子函數
- 7.5 運行時配置
- 7.5.1 全局變量
- 7.5.2 ini配置
- 7.6 函數
- 7.6.1 內部函數注冊
- 7.6.2 函數參數解析
- 7.6.3 引用傳參
- 7.6.4 函數返回值
- 7.6.5 函數調用
- 7.7 zval的操作
- 7.7.1 新生成各類型zval
- 7.7.2 獲取zval的值及類型
- 7.7.3 類型轉換
- 7.7.4 引用計數
- 7.7.5 字符串操作
- 7.7.6 數組操作
- 7.8 常量
- 7.9 面向對象
- 7.9.1 內部類注冊
- 7.9.2 定義成員屬性
- 7.9.3 定義成員方法
- 7.9.4 定義常量
- 7.9.5 類的實例化
- 7.10 資源類型
- 7.11 經典擴展解析
- 7.8.1 Yaf
- 7.8.2 Redis
- 第8章 命名空間
- 8.1 概述
- 8.2 命名空間的定義
- 8.2.1 定義語法
- 8.2.2 內部實現
- 8.3 命名空間的使用
- 8.3.1 基本用法
- 8.3.2 use導入
- 8.3.3 動態用法
- 附錄
- break/continue按標簽中斷語法實現
- defer推遲函數調用語法的實現
- 一起線上事故引發的對PHP超時控制的思考