溫馨提示×

溫馨提示×

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

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

PHP引用的案例分析

發(fā)布時(shí)間:2020-11-03 11:29:54 來源:億速云 閱讀:226 作者:小新 欄目:編程語言

PHP引用的案例分析?這個(gè)問題可能是我們?nèi)粘W(xué)習(xí)或工作經(jīng)常見到的。希望通過這個(gè)問題能讓你收獲頗深。下面是小編給大家?guī)淼膮⒖純?nèi)容,讓我們一起來看看吧!

最近在關(guān)注「PHP 引用」這一話題,看過不少深度文章,對 PHP 里的「引用」有了更深的理解。

0x00

首先看如下代碼:

$foo['hello'] = '0';
$bar = &$foo['hello']; // 引用!
$tipi = $foo;
$tipi['hello'] = '1';

print_r($foo);

問:輸出 0 還是輸出 1?答案是 1。

原理何在?

PHP 內(nèi)核使用 zval 結(jié)構(gòu)存儲變量,在 PHP 代碼里,我們利用 xdebug_debug_zval 函數(shù)一探究竟。

修改如上代碼:

$foo['hello'] = '0';

xdebug_debug_zval('foo');
$bar = &$foo['hello']; // 引用!
xdebug_debug_zval('foo');

$tipi = $foo;
$tipi['hello'] = '1';

print_r($foo);

輸出如下:

foo: (refcount=1, is_ref=0)=array ('hello' => (refcount=1, is_ref=0)='0')
foo: (refcount=1, is_ref=0)=array ('hello' => (refcount=2, is_ref=1)='0')

$foo['hello'] 從非引用變量(is_ref=0)變?yōu)橐米兞浚?code>is_ref=1),而引用計(jì)數(shù)則為 refcount=2。

為什么會這樣?

根據(jù) PHP: 引用做什么 - Manual 的解釋:

$a =& $b; 這意味著 $a 和 $b 指向了同一個(gè)變量。

$a 和 $b 在這里是完全相同的,這并不是 $a 指向了 $b 或者相反,而是 $a 和 $b 指向了同一個(gè)地方。

結(jié)合我們的例子,也就是說,當(dāng) $bar = &$foo['hello']; 執(zhí)行時(shí),$bar$foo['hello'] 都成為了「引用變量」,且它們「指向了同一個(gè)地方」。

那么當(dāng)我們復(fù)制此數(shù)組時(shí),也復(fù)制了它 hello 元素的引用;當(dāng) $tipi['hello'] = '1'; 執(zhí)行時(shí),就修改了 tipi['hello'] 、$foo['hello'] 以及 $bar 所指向的「同一個(gè)地方」。

于是,$foo['hello'] 的值理所當(dāng)然地成為了 1。

0x01

略有深入引用的 PHPer 應(yīng)該都試過這種語法:

for ($list as &$value) {
    $value = 'foo';
}

PHP 在控制結(jié)構(gòu)后不會回收變量,此處不多講解;于是剛剛的坑,其實(shí)可以延伸一下。

$foo['hello'] = '0';
$foo['world'] = 'A';
foreach($foo as &$value) { // 引用!
    // Do nothing.
}
$tipi = $foo;
$tipi['hello'] = '1';
$tipi['world'] = 'B';
print_r($foo);

此處輸出如下:

Array
(
    [hello] => 0
    [world] => B
)

hello 正常,而 world 被修改為 B!原因可以結(jié)合 xdebug_debug_zval 函數(shù)自行探究。

所以,隨手 unset($value); 是個(gè)好習(xí)慣。

0x02

其實(shí)引用也不全是坑。好處還是大大地有。

舉例:

$catList = [
    '1' => ['id' => 1, 'name' => '顏色', 'parent_id' => 0],
    '2' => ['id' => 2, 'name' => '規(guī)格', 'parent_id' => 0],
    '3' => ['id' => 3, 'name' => '白色', 'parent_id' => 1],
    '4' => ['id' => 4, 'name' => '黑色', 'parent_id' => 1],
    '5' => ['id' => 5, 'name' => '大', 'parent_id' => 2],
    '6' => ['id' => 6, 'name' => '小', 'parent_id' => 2],
    '7' => ['id' => 7, 'name' => '黃色', 'parent_id' => 1],
];

如何實(shí)現(xiàn)將如上順序表轉(zhuǎn)換為層級樹?

過去,或者說通常情況下我們首先想到的是遞歸回溯。

不過,利用 PHP 的引用特性,可以將時(shí)間復(fù)雜度降低到 O(n)

$treeData = [];
foreach ($catList as $item) {
    if (isset($catList[$item['parent_id']]) && !empty($catList[$item['parent_id']])) {
        // 子分類
        $catList[$item['parent_id']]['children'][] = &$catList[$item['id']];
    } else {
        // 一級分類
        $treeData[] = &$catList[$item['id']];
    }
}

var_export($treeData);

感謝各位的閱讀!看完上述內(nèi)容,你們對PHP引用的案例分析大概了解了嗎?希望文章內(nèi)容對大家有所幫助。如果想了解更多相關(guān)文章內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI