成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

PHP_底層分析

tomlingtm / 1475人閱讀

摘要:將會(huì)產(chǎn)生強(qiáng)制分裂結(jié)構(gòu)體結(jié)構(gòu)體引用數(shù)組時(shí)的一些奇怪現(xiàn)象引用數(shù)組時(shí)的怪現(xiàn)象數(shù)組不會(huì)比較細(xì)致的檢查,多維數(shù)組存在。因此,判斷的時(shí)候,只會(huì)判斷外面一層的結(jié)構(gòu)體。中底層都離不開表。底層所有的變量都是放在中。

PHP編譯特點(diǎn)

編譯型語言

對于C語言,C++,編譯成機(jī)器碼(二進(jìn)制)來運(yùn)行。
Java語言,把.java 編譯成.class, 稱為bytecode(字節(jié)碼),由jvm來運(yùn)行

解釋型語言

解釋器解釋執(zhí)行。 典型的如: linux shell

解釋器逐行來執(zhí)行命令

PHP執(zhí)行

PHP是先編譯后執(zhí)行

PHP稍有特殊,雖然是一個(gè)腳本語言,但不是靠解釋器解釋。而是zend虛擬機(jī)執(zhí)行,屏蔽了操作系統(tǒng)的區(qū)別。

PHP代碼編譯成 opcode,由zend虛擬機(jī)來執(zhí)行opcode。

但是opcode ,PHP腳本一結(jié)束,opcode就清除了。

opcode 能否緩存

PHP本身不支持,但是apc,xcache等加速器,實(shí)現(xiàn)了這樣的效果。

變量的底層實(shí)現(xiàn)

PHP底層是C語言來實(shí)現(xiàn)的,C語言是強(qiáng)類型,而PHP是弱類型語言,是如何實(shí)現(xiàn)的

PHP的源碼包:

|__ ext
|__ main
|__ pear
|__ sapi
|__ tests
|__ TSRM
|__ Zend
|__ .gdbinit

最核心的是Zend,這是zend虛擬的實(shí)現(xiàn)。包括棧,數(shù)據(jù)類型,編譯器等.
最重要的main,PHP的一些內(nèi)建函數(shù),最重要的函數(shù)都在該目錄下.
最大的一個(gè)目錄 ext, PHP的擴(kuò)展.

PHP的大部分功能,都是以extenstion形式來完成的。
如果自身開發(fā)了一個(gè)擴(kuò)展,也放入ext目錄下。

弱類型語言變量的實(shí)現(xiàn)

/* zend.h  */
struct _zval_struct {
    zvalue_value value;  /* 值 */
    zend_uint refcount__gc;
    zend_uchar type; /* 活動(dòng)類型 */
    zend_uchar is_ref__gc;    
}

PHP中的一個(gè)變量,zend虛擬機(jī)中,使用的是 _zval_struct 的結(jié)構(gòu)體來描述,變量的值也是一個(gè)就結(jié)構(gòu)體來描述.

_zval_struct的結(jié)構(gòu)體是由 四個(gè)字段/域 (可以理解成關(guān)聯(lián)數(shù)組)

zvalue_value value; /* 值 */

PHP變量的值,存儲(chǔ)這個(gè)字段中。

具體存儲(chǔ)的位置:

/* value 值 是一個(gè) 聯(lián)合 */
/* zend.h */
typedef union _zval_value {
    long lval; /* long value */
    double dval; /* double value */
    struct {
        char * val;
        int len;
    } str;
    HashTable *ht; /* hash table 指針 */
    zend_object_value obj;
} zvalue_value;

Zend對變量的表示

zend實(shí)現(xiàn)了 zval結(jié)構(gòu)體

{
    value: [聯(lián)合體] /* 聯(lián)合體的內(nèi)容可能是C語言中的long,double,hashtable(*ht),obj, 聯(lián)合體只能是其中一種類型,是一個(gè)枚舉 */
    type: 變量類型 , /* IS_NULL,IS_BOOL,IS_STRING, IS_LONG,IS_DOUBLE,IS_ARRAY,IS_OBJECT,IS_RESOURCE */
    refcount_gc
    is_ref_gc 
}

C語言中類型對應(yīng)PHP中的數(shù)據(jù)類型:

long -> int
double -> double
hashtable -> array
struct -> string
obj -> object

例如:

$a = 3;
{
    value: [long lval = 3]
    type: IS_LONG
}


$a = 3.5;
{
    value: [double dval = 3.5]
    type: IS_DOUBLE
}
變量類型的實(shí)現(xiàn)

zend_uchar type; /* 活動(dòng)類型 */

可以根據(jù)上下文環(huán)境來強(qiáng)制轉(zhuǎn)換。
例如:需要echo 的時(shí)候 就轉(zhuǎn)換成 string
需要加減運(yùn)算就 轉(zhuǎn)換成 int

PHP 中有8中數(shù)據(jù)類型,為什么zval->value 聯(lián)合體中,只有5中 ?
1: NULL,直接 zval->type = IS_NULL, 就可以表示,不必設(shè)置 value 的值。
2:BOOL, zval->type = IS_BOOL. 再設(shè)置 zval.value.lval = 1/0; (C語言中沒有布爾值,都是通過1,0,來表示)
3: resource ,資源型,往往是服務(wù)器上打開一個(gè)接口,如果 文件讀取接口。 zval->type = IS_RESOURCE, zval->type.lval = 服務(wù)器上打開的接口編號。

struct {
    char * val;
    int len;
} str;

PHP中,字符串類型,長度是已經(jīng)緩存的,調(diào)用strlen時(shí),系統(tǒng)可以直接返回其長度,不需要計(jì)算。

$b = "hello";

/**
 * 
 * {
 *     union_zvalue {
 *      // 字符串的指針
 *         struct{
 *             char: "hello";
 *             len: 5 
 *         } str;
 *     }
 *     type: IS_STRING;
 *  refcount_gc: 1,
 *  is_ref_gc: 0 
 * }
 * 
 */
 
//在PHP中字符串的長度,是直接體現(xiàn)在其結(jié)構(gòu)體中,所以調(diào)用strlen(); 速度非???,時(shí)間復(fù)雜度為0(1)

echo strlen($b);

符號表

符號表symbol_table,變量的花名冊

符號表是什么?

符號表示一張哈希表(哈希結(jié)構(gòu)理解成關(guān)聯(lián)數(shù)組)
里面存儲(chǔ)了變量名-> 變量zval結(jié)構(gòu)體的地址

struct _zend_executor_globals {
    ...
    ...
    HashTable * active_symbol_table /* 活動(dòng)符號表 */
    HashTable symbol_table /* 全局符號表 */
    HashTable included_files; /* files already included */
}

// 變量花名冊
$a = 3;
$b = 1.223;
$c = "hello";

/**
 * 
 * 生成了3個(gè)結(jié)構(gòu)體
 * 同時(shí),全局符號表,中多了三條記錄
 * 
 * a ---> 0x123 ---> 結(jié)構(gòu)體 { 3 }
 * b ---> 0x21a ---> 結(jié)構(gòu)體 { 1.223 }
 * c ---> 0x1A0 ---> 結(jié)構(gòu)體 { hello }
 *
 */
 
 // 變量聲明 
 // 第一:結(jié)構(gòu)體生成
 // 第二:符號表中多了記錄,變量的花名冊
 // 第三:指向結(jié)構(gòu)體 
傳值賦值

傳值賦值發(fā)生了什么

在傳值賦值時(shí):
以:$a = 3; $b = $a;為例:
并沒有再次產(chǎn)生結(jié)構(gòu)體,而是2個(gè)變量共用1個(gè)結(jié)構(gòu)體
此時(shí),2個(gè)變量,指向同1個(gè)結(jié)構(gòu)體
refcount_gc 值為 2 (如果沒有指針指引,會(huì)有垃圾回收機(jī)制清除)

寫時(shí)復(fù)制

cow寫時(shí)復(fù)制特性

$a = 3;
$b = $a;

/**
 * 
 * 是否產(chǎn)生了2 個(gè)結(jié)構(gòu)體?
 * 不是,共用1個(gè), refcount_gc = 2;
 *  
 */

$b = 5;

echo $a, $b; // 3, 5
// $a,$b 指向同一個(gè)結(jié)構(gòu)體,那么,修改$b或$a,對方會(huì)不會(huì)受干擾 ? 沒有干擾到對方。具有寫時(shí)復(fù)制的特性 

如果有一方修改,將會(huì)造成結(jié)構(gòu)體的分裂

結(jié)構(gòu)體一開始共用,到某一方要修改值時(shí),才分裂。這種特性稱為:COW 。Copy On Write。

引用賦值

引用賦值發(fā)生了什么

當(dāng)引用賦值時(shí),雙方共用一個(gè)結(jié)構(gòu)體(is_ref_gc=1)

關(guān)系圖例展示:

強(qiáng)制分裂
1 的過程中(從0到1,表示想引用變量)。refcount_gc>1。多個(gè)變量共享一個(gè)變量值。將會(huì)產(chǎn)生強(qiáng)制分裂
/**
 * // $a $c 結(jié)構(gòu)體 
 *    {
 *    value: 3;
 *    type: IS_LONG;
 *    refcount_gc: 2;  
 *    is_ref_gc: 1; 
 * } 
 * 
 * // $b 結(jié)構(gòu)體
 * {
 *    value: 3;
 *    type: IS_LONG;
 *    refcount_gc: 1;  
 *    is_ref_gc: 0; 
 * }
 *  
 */      

$c = 5;
// a c
/**
 * value: 5
 * type: IS_LONG; 
 * refcount_gc: 2;
 * is_ref_gc: 1;
 */
 
 // b
/**
 * value: 3
 * type: IS_LONG;
 * refcount_gc: 1;
 * is_ref_gc: 0;
 */    

echo $a, $b, $c; // 5 , 3 , 5 

引用數(shù)組時(shí)的一些奇怪現(xiàn)象

// 引用數(shù)組時(shí)的怪現(xiàn)象
    
$arr = array(0, 1, 2, 3);

$tmp = $arr;

$arr[1] = 11;

echo $tmp[1]; // 1

// 數(shù)組不會(huì)比較細(xì)致的檢查,多維數(shù)組存在。 因此,判斷的時(shí)候,只會(huì)判斷外面 一層的 結(jié)構(gòu)體。

數(shù)組不會(huì)比較細(xì)致的檢查

// 先 引用 后 賦值
$arr = array(0, 1, 2, 3);

$x = &$arr[1];

$tmp = $arr;

$arr[1] = 999;

echo $tmp[1]; // 999 . hash表中的zvalue結(jié)構(gòu)體中會(huì)變成引用類型。  // 只去關(guān)注外面一層結(jié)構(gòu)體,而不去關(guān)注 hash表中的值。


echo "
"; // 先賦值,后引用 $arr = array(0, 1, 2, 3); $tmp = $arr; $x = &$arr[1]; $arr[1] = 999; echo $tmp[1]; // 1
循環(huán)數(shù)組

循環(huán)數(shù)組時(shí)的怪現(xiàn)象

// 循環(huán)數(shù)組時(shí)的怪現(xiàn)象
$arr = array(0, 1, 2, 3);

foreach ( $arr as $v ) {
    
}

var_dump(current($arr));  // 數(shù)組指針停留在數(shù)組結(jié)尾處, 取不到值. false

echo "
"; $arr = array(0, 1, 2, 3); foreach ( $arr as $val=>$key ) { // foreach 使用的 $arr 是 $arr的副本. $arr[$key] = $val; // 修改之后,就會(huì)產(chǎn)生分裂。 foreach 遍歷的是 $arr 的副本。 但是原數(shù)組的指針已經(jīng)走了一步. } var_dump(current($arr)); // 1

$arr = array("a", "b", "c", "d");

foreach ( $arr as &$val ) {  // 該foreach 會(huì)導(dǎo)致 $val = &$arr[3];
    
}

foreach ( $arr as $val ) {
    print_r($arr);
    echo "
"; } // 兩個(gè)問題: // 數(shù)組使用時(shí),要慎用引用。 // foreach 使用后,不會(huì)把數(shù)組的內(nèi)部指針重置, 使用數(shù)組時(shí),不要假想內(nèi)部指針指向數(shù)組頭部. 也可以在foreach 之后 reset(); 指針。
符號表與作用域

當(dāng)執(zhí)行到函數(shù)時(shí),會(huì)生成函數(shù)的“執(zhí)行環(huán)境結(jié)構(gòu)體”,包含函數(shù)名,參數(shù),執(zhí)行步驟,所在的類(如果是方法),以及為這個(gè)函數(shù)生成一個(gè)符號表。
符號表統(tǒng)一放在棧上,并把a(bǔ)ctive_symbol_table指向剛產(chǎn)生的符號表。

// Zend/zend_compiles.h 文件中

// 源碼:
struct _zend_execute_data {
    struct _zend_op *opline;
    zend_function_state function_state;
    zend_op_array *op_array;
    zval *object;
    HashTable *symbol_table;
    struct _zend_execute_data *prev_execute_data;
    zval *old_error_reporting;
    zend_bool nested;
    zval **original_return_value;
    zend_class_entry *current_scope;
    zend_class_entry *current_called_scope;
    zval *current_this;
    struct _zend_op *fast_ret; /* used by FAST_CALL/FAST_RET (finally keyword) */
    zval *delayed_exception;
    call_slot *call_slots;
    call_slot *call;
};

// 簡化:

struct _zend_execute_data {
    ...
    zend_op_array *op_array;     // 函數(shù)的執(zhí)行步驟. 如果是函數(shù)調(diào)用。是函數(shù)調(diào)用的后的opcode
    HashTable *symbol_table; // 此函數(shù)的符號表地址
    zend_class_entry *current_scope; // 執(zhí)行當(dāng)前作用域
    zval * current_this;  // 對象 調(diào)用 this綁定 
    zval * current_object;  // object 的指向
    ...
}

一個(gè)函數(shù)調(diào)用多次,會(huì)有多少個(gè)*op_array ?
一個(gè)函數(shù)產(chǎn)生 一個(gè)*op_array. 調(diào)用多次,會(huì)產(chǎn)生多個(gè) 環(huán)境結(jié)構(gòu)體, 會(huì)依次入棧,然后順序執(zhí)行。
調(diào)用多少次,就會(huì)入棧多少次。不同的執(zhí)行環(huán)境,靠 唯一的 *op_array 來執(zhí)行。

函數(shù)什么時(shí)候調(diào)用, 函數(shù)編譯后的 opcode 什么時(shí)候執(zhí)行。

$age = 23;

function t() {
    $age = 3;
    echo $age;
}

t();

/**
 * t 函數(shù) 在執(zhí)行時(shí),根據(jù)函數(shù)的參數(shù),局部變量等,生成一個(gè)執(zhí)行環(huán)境結(jié)構(gòu)體。
 * 結(jié)構(gòu)體 入棧,函數(shù)編譯后的 opcode, 稱為 op_array (就是執(zhí)行邏輯)。開始執(zhí)行, 以入棧的環(huán)境結(jié)構(gòu)體為環(huán)境來執(zhí)行。
 * 并生成此函數(shù)的 符號表, 函數(shù)尋找變量, 就在符號表中尋找。即局部變量。(一個(gè)環(huán)境結(jié)構(gòu)體,就對應(yīng)一張符號表)
 * 
 * 
 * 注意: 函數(shù)可能調(diào)用多次。棧中可能有某函數(shù)的多個(gè)執(zhí)行環(huán)境 入棧。但是 op_array 只有一個(gè)。
 * 
 */
靜態(tài)變量

靜態(tài)變量的實(shí)現(xiàn)

// Zend/zend_compile.h  
struct _zend_op_array {
    /* Common elements */
    zend_uchar type;
    const 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;
    /* END of common elements */

    zend_uint *refcount;

    zend_op *opcodes;
    zend_uint last;

    zend_compiled_variable *vars;
    int last_var;

    zend_uint T;

    zend_uint nested_calls;
    zend_uint used_stack;

    zend_brk_cont_element *brk_cont_array;
    int last_brk_cont;

    zend_try_catch_element *try_catch_array;
    int last_try_catch;
    zend_bool has_finally_block;

    /* static variables support */
    HashTable *static_variables;

    zend_uint this_var;

    const char *filename;
    zend_uint line_start;
    zend_uint line_end;
    const char *doc_comment;
    zend_uint doc_comment_len;
    zend_uint early_binding; /* the linked list of delayed declarations */

    zend_literal *literals;
    int last_literal;

    void **run_time_cache;
    int  last_cache_slot;

    void *reserved[ZEND_MAX_RESERVED_RESOURCES];
};

// 簡化
struct _zend_op_array {
    ... 
    HashTable *static_variables;    // 靜態(tài)變量
    ...
}

編譯后的 op_array 只有一份。 靜態(tài)變量并沒有存儲(chǔ)在符號表(symbol_table)中.而是存放在op_array中。

function t() {
    
    static $age = 1;
    
    return $age += 1;
    
}

echo t();
echo t();
echo t();

// 靜態(tài)變量 不再和 執(zhí)行的結(jié)構(gòu)體, 也不再和 入棧的符號表有關(guān)。

常量
// Zend/zend_constants.h
// 常量結(jié)構(gòu)體 
typedef struct _zend_constant {
    zval value; // 變量結(jié)構(gòu)體
    int flags; // 標(biāo)志,是否大小寫敏感等
    char *name; // 常量名
    uint name_len; // 
    int module_number; // 模塊名
} zend_constant;
define函數(shù)的實(shí)現(xiàn)

define函數(shù)當(dāng)然是 調(diào)用zend_register_constant聲明的常量
具體如下:Zend/zend_builtin_functions.c

// 源碼:

ZEND_FUNCTION(define)
{
    char *name;
    int name_len;
    zval *val;
    zval *val_free = NULL;
    zend_bool non_cs = 0;
    int case_sensitive = CONST_CS;
    zend_constant c;

    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "sz|b", &name, &name_len, &val, &non_cs) == FAILURE) {
        return;
    }

    if(non_cs) {
        case_sensitive = 0;
    }

    /* class constant, check if there is name and make sure class is valid & exists */
    if (zend_memnstr(name, "::", sizeof("::") - 1, name + name_len)) {
        zend_error(E_WARNING, "Class constants cannot be defined or redefined");
        RETURN_FALSE;
    }

repeat:
    switch (Z_TYPE_P(val)) {
        case IS_LONG:
        case IS_DOUBLE:
        case IS_STRING:
        case IS_BOOL:
        case IS_RESOURCE:
        case IS_NULL:
            break;
        case IS_OBJECT:
            if (!val_free) {
                if (Z_OBJ_HT_P(val)->get) {
                    val_free = val = Z_OBJ_HT_P(val)->get(val TSRMLS_CC);
                    goto repeat;
                } else if (Z_OBJ_HT_P(val)->cast_object) {
                    ALLOC_INIT_ZVAL(val_free);
                    if (Z_OBJ_HT_P(val)->cast_object(val, val_free, IS_STRING TSRMLS_CC) == SUCCESS) {
                        val = val_free;
                        break;
                    }
                }
            }
            /* no break */
        default:
            zend_error(E_WARNING,"Constants may only evaluate to scalar values");
            if (val_free) {
                zval_ptr_dtor(&val_free);
            }
            RETURN_FALSE;
    }
    
    c.value = *val;
    zval_copy_ctor(&c.value);
    if (val_free) {
        zval_ptr_dtor(&val_free);
    }
    c.flags = case_sensitive; /* non persistent */
    c.name = str_strndup(name, name_len);
    if(c.name == NULL) {
        RETURN_FALSE;
    }
    c.name_len = name_len+1;
    c.module_number = PHP_USER_CONSTANT;
    if (zend_register_constant(&c TSRMLS_CC) == SUCCESS) {
        RETURN_TRUE;
    } else {
        RETURN_FALSE;
    }
}

// 關(guān)鍵代碼:

c.value = *val;
zval_copy_ctor(&c.value);
if (val_free) {
    zval_ptr_dtor(&val_free);
}
c.flags = case_sensitive; /* 大小寫敏感 */
c.name = str_strndup(name, name_len);
if(c.name == NULL) {
    RETURN_FALSE;
}
c.name_len = name_len+1;
c.module_number = PHP_USER_CONSTANT; /* 用戶定義常量 */
if (zend_register_constant(&c TSRMLS_CC) == SUCCESS) {
    RETURN_TRUE;
} else {
    RETURN_FALSE;
}

常量就一個(gè)符號(哈希)表. 都使用一個(gè)符號表。所以全局有效。

常量的生成

int zend_register_constant(zend_constant *c TSRMLS_DC) {
    ...
    ...
    zend_hash_add(EG(zend_constants), name, c->name_len, (vaid*)c,sizeof(zend_constant, NULL) == FAILURE);
    ...
    ...
}

對象定義常量

class Dog {
    
    public $name = "kitty";
    
    public function __toString () {
        return $this->name;
    }
    
}

$dog = new Dog();


define("DOG", $dog);

print_r(DOG);

/**
 * define 值為對象時(shí),會(huì)把對象裝成標(biāo)量來存儲(chǔ),需要類有 __toString魔術(shù)方法
 */    
 
對象

對象的底層實(shí)現(xiàn)

Zend/zend.h

struct _zval_struct {
    /* Variable information */
    zvalue_value value;        /* value */
    zend_uint refcount__gc;
    zend_uchar type;    /* active type */
    zend_uchar is_ref__gc;
};

// zvalue
typedef union _zvalue_value {
    long lval;                    /* long value */
    double dval;                /* double value */
    struct {
        char *val;
        int len;
    } str;
    HashTable *ht;                /* hash table value */
    zend_object_value obj;
    zend_ast *ast;
} zvalue_value;

// 在 zend.h 中 查看到 `zend_object_value obj;`  是以zend_object_value 定義. 在Zend/zend_types.h 文件中繼續(xù)查看

// Zend/zend_types.h

定義zend_object_value 結(jié)構(gòu)體

typedef struct _zend_object_value {
    zend_object_handle handle;
    const zend_object_handlers *handlers;
} zend_object_value;

通過new出來的對象,返回的是什么。是zend_object_value. 并不是真正的對象,而是對象的指針。

返回的 handle再次指向?qū)ο蟆?/p>

每次new一個(gè)對象,對象就存入一張hash表中。(形象的稱之為對象池)

對象存儲(chǔ)時(shí)的特點(diǎn):

// 對象

class Dog {
    public $leg = 4;
    public $wei = 20;
}

$dog = new Dog();

// $dog 是一個(gè)對象么?
// 嚴(yán)格說,并不是對象.
/**
 * {
 *     handle --指向-->  [hash表 {leg: 4, wei: 20}] // hash表中存在 對象    
 * }
 */
 
$d2 = $dog;

$d2->leg = 5;

echo $dog->leg, "`", $d2->leg; // 5`5

// 對象并不是 引用賦值. 主要原因  zval 結(jié)構(gòu)體 是再次指向一個(gè)hash表中的 對象池
$d2 = false;

echo $dog->leg; // 5
    

內(nèi)存分層

內(nèi)存管理與垃圾回收

PHP封裝了對系統(tǒng)內(nèi)存的請求
不要直接使用malloc直接請求內(nèi)存

PHP函數(shù)需要內(nèi)存的時(shí)候,是通過emalloc,efree.
emalloc,efree向 mm_heap索要空間。

zend 中底層都離不開hash表。PHP中的HashTable太強(qiáng)大。

PHP 底層 所有的變量都是 放在 zend_mm_heap 中。 然后通過 各自的hash表來指向或跟蹤。

zend虛擬機(jī)的運(yùn)行原理

PHP語法實(shí)現(xiàn)

Zend/zend_language_scanner.l
Zend/zend_language_parser.y

OPcode編譯

Zend/zend.compile.c

執(zhí)行引擎

Zend/zend_vm_*
Zend/zend_execute.c

以apache模塊運(yùn)行時(shí)的流程

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/30378.html

相關(guān)文章

  • PHP底層分析:關(guān)于強(qiáng)制分裂

    摘要:前面寫過一篇底層分析關(guān)于寫時(shí)復(fù)制今天來講講關(guān)于強(qiáng)制分裂的知識,簡單來說,強(qiáng)制分裂就是在引用變量主動(dòng)賦值前,該變量傳值賦值過,就會(huì)發(fā)生強(qiáng)制分裂。實(shí)際開發(fā)基本用不到這層原理,但在面試中強(qiáng)制分裂通常會(huì)和寫時(shí)復(fù)制一起考。 學(xué)習(xí)需要知其然而知其所以然,PHP底層相關(guān)就是這類知識。 前面寫過一篇《PHP底層分析:關(guān)于寫時(shí)復(fù)制(cow)》:https://segmentfault.com/a/119...

    劉明 評論0 收藏0
  • PHP底層分析:關(guān)于寫時(shí)復(fù)制(cow)

    摘要:可以看到,該結(jié)構(gòu)體存儲(chǔ)了關(guān)于變量值,有幾個(gè)變量指向該結(jié)構(gòu)體,變量類型,是否為引用變量等信息。這個(gè)就是寫時(shí)復(fù)制,在作怪,他沒有在賦值的時(shí)候就分裂成兩個(gè)結(jié)構(gòu)體,而是在我們改寫其中一個(gè)變量時(shí)發(fā)生效果,屬于一種慢復(fù)制也稱慢分裂。 想要走到技術(shù)的天花板,那么學(xué)習(xí)過程中在于知其然且知其所以然。 今天我們來討論一下PHP底層的寫時(shí)復(fù)制(也稱寫時(shí)分裂)。 首先我們先來看看一段代碼:showImg(ht...

    widuu 評論0 收藏0
  • foreach遍歷過程中的奇怪現(xiàn)象(PHP5)

    摘要:中基礎(chǔ)中的三大坑,遍歷,引用機(jī)制,數(shù)組。今天我們在講講中的一些奇怪現(xiàn)象。本文適合有一定基礎(chǔ)的。運(yùn)行流程共用一個(gè)結(jié)構(gòu)體開始遍歷數(shù)組,進(jìn)行判斷,拷貝數(shù)組是一個(gè)新的結(jié)構(gòu)體,操作的是新的結(jié)構(gòu)體。那么遍歷數(shù)組時(shí),全程與原數(shù)組無關(guān)。 PHP中基礎(chǔ)中的三大坑,foreach遍歷,引用機(jī)制&,數(shù)組。 今天我們在講講foreach中的一些奇怪現(xiàn)象。 在講解之前,可以先看看我其他相關(guān)的文章,屬于同一個(gè)大的...

    kgbook 評論0 收藏0
  • php底層原理之函數(shù)

    摘要:但是到底是如何找到對應(yīng)的函數(shù)的呢今天,我們來一起尋找答案函數(shù)分類首先,我們先回顧一下的函數(shù)分類,函數(shù)包含用戶自定義函數(shù)內(nèi)部函數(shù)匿名函數(shù)等多種類型。用戶自定義函數(shù)和內(nèi)部函數(shù)編譯完成后會(huì)將函數(shù)名注冊到全局函數(shù)列表中。 對于PHPer而言,我們通常會(huì)把常用的功能封裝成一個(gè)函數(shù)來進(jìn)行復(fù)用,以提升開發(fā)效率。但是php到底是如何找到對應(yīng)的函數(shù)的呢?今天,我們來一起尋找答案~ 函數(shù)分類 首先,我們先...

    rose 評論0 收藏0
  • PHP的生成器

    摘要:它最簡單的調(diào)用形式看起來像一個(gè)申明,不同之處在于普通會(huì)返回值并終止函數(shù)的執(zhí)行,而會(huì)返回一個(gè)值給循環(huán)調(diào)用此生成器的代碼并且只是暫停執(zhí)行生成器函數(shù)。 0x01 寫在前面 本文主要介紹: Generator的簡單用法。 Generator的底層實(shí)現(xiàn)。 本文比較長,可能會(huì)耗費(fèi)你比較多的時(shí)間。如果你比較了解Generator的用法,僅想了解底層實(shí)現(xiàn),可以直接跳到底層實(shí)現(xiàn)部分。 本文分析的PH...

    LMou 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<