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

資訊專欄INFORMATION COLUMN

搞定PHP面試 - 變量的引用賦值與傳值賦值

liangdas / 2571人閱讀

摘要:比如變量的多次賦值函數(shù)參數(shù)傳遞,并在函數(shù)體內(nèi)修改實(shí)參等。引用賦值定義一個變量定義變量,將變量的引用賦給對進(jìn)行修改輸出結(jié)果定義一個變量定義變量,將變量的引用賦給對進(jìn)行修改二使用查看變量的引用情況用于顯示變量的信息。

一、使用 memory_get_usage() 查看PHP內(nèi)存使用量 1. 傳值賦值
// 定義一個變量
$a = range(0, 10000);
var_dump(memory_get_usage());

// 定義變量b,將a變量的值賦值給b
$b = $a;
var_dump(memory_get_usage());

// 對a進(jìn)行修改
// COW: Copy-On-Write
$a = range(0, 10000);
var_dump(memory_get_usage());

輸出結(jié)果:

int(989768)
int(989856)
int(1855608)
定義一個變量 $a = range(0, 10000);

$b = $a;

對a進(jìn)行修改 $a = range(0, 10000);

PHP寫時復(fù)制機(jī)制(Copy-on-Write,也縮寫為COW)

顧名思義,就是在寫入時才真正復(fù)制一份內(nèi)存進(jìn)行修改。
COW最早應(yīng)用在Unix系統(tǒng)中對線程與內(nèi)存使用的優(yōu)化,后面廣泛的被使用在各種編程語言中,如C++的STL等。
在PHP內(nèi)核中,COW也是主要的內(nèi)存優(yōu)化手段。
在通過變量賦值的方式賦值給變量時,不會申請新內(nèi)存來存放新變量的值,而是簡單的通過一個計(jì)數(shù)器來共用內(nèi)存。只有在其中的一個引用指向變量的值發(fā)生變化時,才申請新空間來保存值內(nèi)容,以減少對內(nèi)存的占用。
在很多場景下PHP都使用COW進(jìn)行內(nèi)存的優(yōu)化。比如:變量的多次賦值、函數(shù)參數(shù)傳遞,并在函數(shù)體內(nèi)修改實(shí)參等。

2. 引用賦值
// 定義一個變量
$a = range(0, 10000);
var_dump(memory_get_usage());

// 定義變量b,將a變量的引用賦給b
$b = &$a;
var_dump(memory_get_usage());

// 對a進(jìn)行修改
$a = range(0, 10000);
var_dump(memory_get_usage());

輸出結(jié)果:

int(989760)
int(989848)
int(989840)
定義一個變量 $a = range(0, 10000);

定義變量b,將a變量的引用賦給b $b = &$a;

對a進(jìn)行修改 $a = range(0, 10000);

二、使用 xdebug_debug_zval() 查看變量的引用情況
xdebug_debug_zval() 用于顯示變量的信息。需要安裝xdebug擴(kuò)展。
1. 傳值賦值
$a = 1;
xdebug_debug_zval("a");

// 定義變量b,把a(bǔ)的值賦值給b
$b = $a;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

// a進(jìn)行寫操作
$a = 2;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

輸出結(jié)果:

a: (refcount=1, is_ref=0)=1
a: (refcount=2, is_ref=0)=1
b: (refcount=2, is_ref=0)=1
a: (refcount=1, is_ref=0)=2
b: (refcount=1, is_ref=0)=1
定義變量 $a = 1;
$a = 1;
xdebug_debug_zval("a");

輸出

a: (refcount=1, is_ref=0)=1

refcount=1 表示該變量指向的內(nèi)存地址的引用個數(shù)變?yōu)?
is_ref=0 表示該變量不是引用

定義變量 $b ,把 $a 的值賦給 $b, $b = $a;
$b = $a;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

輸出

a: (refcount=2, is_ref=0)=1
b: (refcount=2, is_ref=0)=1

refcount=2 表示該變量指向的內(nèi)存地址的引用個數(shù)變?yōu)?
is_ref=0 表示該變量不是引用

對變量 $a 進(jìn)行寫操作 $a = 2;
$a = 2;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

輸出

a: (refcount=1, is_ref=0)=2
b: (refcount=1, is_ref=0)=1

因?yàn)镃OW機(jī)制,對變量 $a 進(jìn)行寫操作時,會為變量 $a 新分配一塊內(nèi)存空間,用于存儲變量 $a 的值。
此時 $a$b 指向的內(nèi)存地址的引用個數(shù)都變?yōu)?。

2. 引用賦值
$a = 1;
xdebug_debug_zval("a");

// 定義變量b,把a(bǔ)的引用賦給b
$b = &$a;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

// a進(jìn)行寫操作
$a = 2;
xdebug_debug_zval("a");
xdebug_debug_zval("b");
a: (refcount=1, is_ref=0)=1
a: (refcount=2, is_ref=1)=1
b: (refcount=2, is_ref=1)=1
a: (refcount=2, is_ref=1)=2
b: (refcount=2, is_ref=1)=2
定義變量 $a = 1;
$a = 1;
xdebug_debug_zval("a");

輸出

a: (refcount=1, is_ref=0)=1

refcount=1 表示該變量指向的內(nèi)存地址的引用個數(shù)變?yōu)?
is_ref=0 表示該變量不是引用

定義變量 $b ,把 $a 的引用賦給 $b$b = &$a;
$b = &$a;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

輸出

a: (refcount=2, is_ref=1)=1
b: (refcount=2, is_ref=1)=1

refcount=2 表示該變量指向的內(nèi)存地址的引用個數(shù)變?yōu)?
is_ref=1 表示該變量是引用

對變量 $a 進(jìn)行寫操作 $a = 2;
$a = 2;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

輸出

a: (refcount=2, is_ref=1)=2
b: (refcount=2, is_ref=1)=2

因?yàn)樽兞?$a 和變量 $b 指向相同的內(nèi)存地址,其實(shí)引用。
對變量 $a 進(jìn)行寫操作時,會直接修改指向的內(nèi)存空間的值,因此變量 $b 的值會跟著一起改變。

三、當(dāng)變量時引用時,unset()只會取消引用,不會銷毀內(nèi)存空間
$a = 1;
$b = &$a;

// unset 只會取消引用,不會銷毀內(nèi)存空間
unset($b);

echo $a;

輸出

1
定義變量 $a ,并將 $a 的引用賦給變量 $b
$a = 1;
$b = &$a;

銷毀 $b
unset($b);

輸出 $a

雖然銷毀的 $b,但是 $a 的引用和內(nèi)存空間依舊存在。

echo $a;

輸出

1
四、php中對象本身就是引用賦值
class Person
{
    public $age = 1;
}

$p1 = new Person;
xdebug_debug_zval("p1");

$p2 = $p1;
xdebug_debug_zval("p1");
xdebug_debug_zval("p2");

$p2->age = 2;
xdebug_debug_zval("p1");
xdebug_debug_zval("p2");
p1: (refcount=1, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }
p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }
實(shí)例化對象 $p1 = new Person;
$p1 = new Person;
xdebug_debug_zval("p1");

輸出

p1: (refcount=1, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }

refcount=1 表示該變量指向的內(nèi)存地址的引用個數(shù)變?yōu)?
is_ref=0 表示該變量不是引用

$p1 賦給 $p2
$p2 = $p1;
xdebug_debug_zval("p1");
xdebug_debug_zval("p2");

輸出

p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }

refcount=2 表示該變量指向的內(nèi)存地址的引用個數(shù)變?yōu)?

$p2 中的屬性 age 進(jìn)行寫操作
$p2->age = 2;
xdebug_debug_zval("p1");
xdebug_debug_zval("p2");

輸出

p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }
p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }

因?yàn)閜hp中對象本身就是引用賦值。對 $p2 中的屬性 age 進(jìn)行寫操作時,會直接修改指向的內(nèi)存空間的值,因此變量 $p1age 屬性的值會跟著一起改變。

五、實(shí)戰(zhàn)例題分析
/**
 * 寫出如下程序的輸出結(jié)果
 *
 * $d = ["a", "b", "c"];
 *
 * foreach($d as $k => $v)
 * {
 *    $v = &$d[$k];
 * }
 * 
 * 程序運(yùn)行時,每一次循環(huán)結(jié)束后變量 $d 的值是什么?請解釋。
 * 程序執(zhí)行完成后,變量 $d 的值是什么?請解釋。
 */
1. 第一次循環(huán) 推算出進(jìn)入 foreach$v、$d[$k] 的值
$k = 0
$v = "a"
$d[$k] = $d[0] = "a"

此時,$v$d[0] 在內(nèi)存中分別開辟了一塊空間

![$v 和 $d[0] 在內(nèi)存中分別開辟了一塊空間](http://md.ws65535.top/xsj/201...

$v = &$d[0] 改變了 $v 指向的內(nèi)存地址
$v = &$d[0]

![$v = &$d[0] 改變了 $val 指向的內(nèi)存地址](http://md.ws65535.top/xsj/201...

第一次循環(huán)后 $d 的值:
["a", "b", "c"]
2. 第二次循環(huán) 進(jìn)入 foreach$v 被賦值為 "b",此時$v指向的內(nèi)存地址與 $d[0] 相同,且為引用,因此 $d[0] 的值被修改為 "b"

$v = "b" => $d[0] = "b"

![$v = ‘b’ => $d[0] = ‘b’](http://md.ws65535.top/xsj/201...

推算出進(jìn)入 foreach$d[$k] 的值
$k = 1
$d[$k] = $d[1] = "b"

![$d[2] = ‘b’](http://md.ws65535.top/xsj/201...

$v = &$d[1] 改變了 $v 指向的內(nèi)存地址
$v = &$d[1]

![$v = &$d[1]](http://md.ws65535.top/xsj/201...

第二次循環(huán)后 $d 的值
["b", "b", "c"]
3. 第三次循環(huán) 進(jìn)入 foreach$v 被賦值為 "c",此時$v指向的內(nèi)存地址與 $d[1] 相同,且為引用,因此 $d[1] 的值被修改為 "c"

$v = "c" => $d[1] = "c"

![$v = ‘c’ => $d[1] = ‘c’](http://md.ws65535.top/xsj/201...

推算出進(jìn)入 foreach$d[$k] 的值
$k = 2
$d[2] = "c"

![$d[2] = ‘c’](http://md.ws65535.top/xsj/201...

$v = &$d[2] 改變了 $v 指向的內(nèi)存地址
$v = &$d[2]

![$v = &$d[2]](http://md.ws65535.top/xsj/201...

第三次循環(huán)后 $d 的值
["b", "c", "c"]
4. 實(shí)測
$d = ["a", "b", "c"];

foreach ($d as $k=>$v)
{
    $v = &$d[$k];
    print_r($d);
}

print_r($d);
輸出:
Array
(
    [0] => a
    [1] => b
    [2] => c
)
Array
(
    [0] => b
    [1] => b
    [2] => c
)
Array
(
    [0] => b
    [1] => c
    [2] => c
)
Array
(
    [0] => b
    [1] => c
    [2] => c
)

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

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

相關(guān)文章

  • 搞定PHP面試 - 變量知識點(diǎn)整理

    摘要:聲明靜態(tài)變量時不能用表達(dá)式的結(jié)果對其賦值正確錯誤使用表達(dá)式的結(jié)果賦值錯誤使用表達(dá)式的結(jié)果賦值靜態(tài)變量與遞歸函數(shù)靜態(tài)變量提供了一種處理遞歸函數(shù)的方法。 一、變量的定義 1. 變量的命名規(guī)則 變量名可以包含字母、數(shù)字、下劃線,不能以數(shù)字開頭。 $Var_1 = foo; // 合法 $var1 = foo; // 合法 $_var1 = foo; // 合法 $Var-1 = foo; /...

    Mertens 評論0 收藏0
  • 搞定PHP面試 - 深入了解引用

    摘要:引用可以被看作是文件系統(tǒng)中的硬鏈接。如果具有引用的數(shù)組被復(fù)制,其值不會解除引用。如果試圖這樣從函數(shù)返回引用,將會報錯,因?yàn)楹瘮?shù)在試圖返回一個表達(dá)式的結(jié)果而不是一個引用的變量。這并不意味著變量內(nèi)容被銷毀了。 1. 什么是引用 在 PHP 中引用是指用不同的名字訪問同一個變量內(nèi)容。PHP 中的變量名和變量內(nèi)容是不一樣的, 因此同樣的內(nèi)容可以有不同的名字。最接近的比喻是 Unix 的文件名和...

    fox_soyoung 評論0 收藏0
  • 一篇文章理解JS數(shù)據(jù)類型、深拷貝和淺拷貝

    摘要:接下來我們進(jìn)入正片數(shù)據(jù)類型六種基本數(shù)據(jù)類型布爾值,和一個表明值的特殊關(guān)鍵字。一種數(shù)據(jù)類型,它的實(shí)例是唯一且不可改變的。在中是沒有方法是可以改變布爾值和數(shù)字的。參考資料深拷貝淺拷貝 前言 筆者最近整理了一些前端技術(shù)文章,如果有興趣可以參考這里:muwoo blogs。接下來我們進(jìn)入正片: js 數(shù)據(jù)類型 六種 基本數(shù)據(jù)類型: Boolean. 布爾值,true 和 false. nu...

    EddieChan 評論0 收藏0
  • 一篇文章理解JS數(shù)據(jù)類型、深拷貝和淺拷貝

    摘要:接下來我們進(jìn)入正片數(shù)據(jù)類型六種基本數(shù)據(jù)類型布爾值,和一個表明值的特殊關(guān)鍵字。一種數(shù)據(jù)類型,它的實(shí)例是唯一且不可改變的。在中是沒有方法是可以改變布爾值和數(shù)字的。參考資料深拷貝淺拷貝 前言 筆者最近整理了一些前端技術(shù)文章,如果有興趣可以參考這里:muwoo blogs。接下來我們進(jìn)入正片: js 數(shù)據(jù)類型 六種 基本數(shù)據(jù)類型: Boolean. 布爾值,true 和 false. nu...

    enda 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<