<ruby id="bdb3f"></ruby>

    <p id="bdb3f"><cite id="bdb3f"></cite></p>

      <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
        <p id="bdb3f"><cite id="bdb3f"></cite></p>

          <pre id="bdb3f"></pre>
          <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

          <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
          <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

          <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                <ruby id="bdb3f"></ruby>

                ??碼云GVP開源項目 12k star Uniapp+ElementUI 功能強大 支持多語言、二開方便! 廣告
                ## 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): ![](https://box.kancloud.cn/02c668bac663543c8753c29fe241b629_406x148.png) ### 2.4.2 全局變量的訪問 與靜態變量的訪問一樣,全局變量也是將原來的值轉換為引用,然后在global導入的作用域內創建一個局部變量指向該引用: ```php global $id; // 相當于:$id = & EG(symbol_table)["id"]; ``` 具體的操作過程不再細講,與靜態變量的處理過程一致,這時示例中局部變量與全局變量的引用情況如下圖。 ![](https://box.kancloud.cn/1b8a3e9cdf810c0eb32cef4ec5f6fd0f_547x283.png) ### 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))); } ... } ```
                  <ruby id="bdb3f"></ruby>

                  <p id="bdb3f"><cite id="bdb3f"></cite></p>

                    <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
                      <p id="bdb3f"><cite id="bdb3f"></cite></p>

                        <pre id="bdb3f"></pre>
                        <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

                        <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
                        <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

                        <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                              <ruby id="bdb3f"></ruby>

                              哎呀哎呀视频在线观看