成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

PHP的垃圾回收機制——引用計數(shù)

開發(fā) 后端
每個php變量存在一個叫"zval"的變量容器中。一個zval變量容器,除了包含變量的類型和值,還包括兩個字節(jié)的額外信息。第一個是"is_ref",是個bool值,用來標識這個變量是否是屬于引用集合(reference set)。

[[206092]]

每個php變量存在一個叫"zval"的變量容器中。一個zval變量容器,除了包含變量的類型和值,還包括兩個字節(jié)的額外信息。***個是"is_ref",是個bool值,用來標識這個變量是否是屬于引用集合(reference set)。通過這個字節(jié),php引擎才能把普通變量和引用變量區(qū)分開來,由于php允許用戶通過使用&來使用自定義引用,zval變量容器中還有一個內(nèi)部引用計數(shù)機制,來優(yōu)化內(nèi)存使用。第二個額外字節(jié)是"refcount",用以表示指向這個zval變量容器的變量(也稱符號即symbol)個數(shù)。所有的符號存在一個符號表中,其中每個符號都有作用域(scope),那些主腳本(比如:通過瀏覽器請求的的腳本)和每個函數(shù)或者方法也都有作用域。

當(dāng)一個變量被賦常量值時,就會生成一個zval變量容器,如下例這樣:

例1 生成一個新的zval容器

  1. <?php  
  2.  
  3. $a = "new string"
  4.  
  5. ?> 

在上例中,新的變量a,是在當(dāng)前作用域中生成的。并且生成了類型為 string 和值為new string的變量容器。在額外的兩個字節(jié)信息中,"is_ref"被默認設(shè)置為 FALSE,因為沒有任何自定義的引用生成。"refcount" 被設(shè)定為 1,因為這里只有一個變量使用這個變量容器. 注意到當(dāng)"refcount"的值是1時,"is_ref"的值總是FALSE. 如果你已經(jīng)安裝了» Xdebug,你能通過調(diào)用函數(shù) xdebug_debug_zval()顯示"refcount"和"is_ref"的值。

例2 顯示zval信息

  1. <?php  
  2.  
  3. xdebug_debug_zval('a'); 
  4.  
  5. ?> 

以上例程會輸出:

  1. a: (refcount=1, is_ref=0)='new string' 

把一個變量賦值給另一變量將增加引用次數(shù)(refcount).

例3 增加一個zval的引用計數(shù)

  1. <?php 
  2.  $a = "new string"
  3.  
  4. $b = $a; 
  5.  
  6. xdebug_debug_zval( 'a' ); 
  7.  
  8. ?> 

以上例程會輸出:

  1. a: (refcount=2, is_ref=0)='new string' 

這時,引用次數(shù)是2,因為同一個變量容器被變量 a 和變量 b關(guān)聯(lián).當(dāng)沒必要時,php不會去復(fù)制已生成的變量容器。變量容器在”refcount“變成0時就被銷毀. 當(dāng)任何關(guān)聯(lián)到某個變量容器的變量離開它的作用域(比如:函數(shù)執(zhí)行結(jié)束),或者對變量調(diào)用了函數(shù) unset()時,”refcount“就會減1,下面的例子就能說明:

例4 減少引用計數(shù)

  1. <?php  
  2.  
  3. $a = "new string"
  4.  
  5. $c = $b = $a; 
  6.  
  7. xdebug_debug_zval( 'a' ); 
  8.  
  9. unset( $b, $c ); 
  10.  
  11. xdebug_debug_zval( 'a' ); 
  12.  
  13. ?> 

以上例程會輸出:

  1. a: (refcount=3, is_ref=0)='new string' 
  2.  
  3. a: (refcount=1, is_ref=0)='new string' 

如果我們現(xiàn)在執(zhí)行 unset($a);,包含類型和值的這個變量容器就會從內(nèi)存中刪除。

復(fù)合類型(Compound Types)

當(dāng)考慮像 array和object這樣的復(fù)合類型時,事情就稍微有點復(fù)雜. 與 標量(scalar)類型的值不同,array和 object類型的變量把它們的成員或?qū)傩源嬖谧约旱姆柋碇小_@意味著下面的例子將生成三個zval變量容器。

例5 Creating a array zval

  1. <?php  
  2.  
  3. $a = array( 'meaning' => 'life''number' => 42 ); 
  4.  
  5. xdebug_debug_zval( 'a' ); 
  6.  
  7. ?> 

以上例程的輸出類似于:

  1. a: (refcount=1, is_ref=0)=array ( 
  2.    'meaning' => (refcount=1, is_ref=0)='life'
  3.    'number' => (refcount=1, is_ref=0)=42 

圖示:

 

一個簡單數(shù)組的zval

這三個zval變量容器是: a,meaning和 number。增加和減少”refcount”的規(guī)則和上面提到的一樣. 下面, 我們在數(shù)組中再添加一個元素,并且把它的值設(shè)為數(shù)組中已存在元素的值:

例6 添加一個已經(jīng)存在的元素到數(shù)組中

  1. <?php  
  2.  
  3. $a = array( 'meaning' => 'life''number' => 42 ); 
  4.  
  5. $a['life'] = $a['meaning']; 
  6.  
  7. xdebug_debug_zval( 'a' ); 
  8.  
  9. ?> 

以上例程的輸出類似于:

  1. a: (refcount=1, is_ref=0)=array ( 
  2.  
  3. 'meaning' => (refcount=2, is_ref=0)='life'
  4.  
  5. 'number' => (refcount=1, is_ref=0)=42, 
  6.  
  7. 'life' => (refcount=2, is_ref=0)='life' 
  8.  

圖示:

 

帶有引用的簡單數(shù)組的zval

從以上的xdebug輸出信息,我們看到原有的數(shù)組元素和新添加的數(shù)組元素關(guān)聯(lián)到同一個"refcount"2的zval變量容器. 盡管 Xdebug的輸出顯示兩個值為'life'的 zval 變量容器,其實是同一個。 函數(shù)xdebug_debug_zval()不顯示這個信息,但是你能通過顯示內(nèi)存指針信息來看到。

刪除數(shù)組中的一個元素,就是類似于從作用域中刪除一個變量. 刪除后,數(shù)組中的這個元素所在的容器的“refcount”值減少,同樣,當(dāng)“refcount”為0時,這個變量容器就從內(nèi)存中被刪除,下面又一個例子可以說明:

例7 從數(shù)組中刪除一個元素

  1. <?php  
  2.  
  3. $a = array( 'meaning' => 'life''number' => 42 ); 
  4.  
  5. $a['life'] = $a['meaning']; 
  6.  
  7. unset( $a['meaning'], $a['number'] ); 
  8.  
  9. xdebug_debug_zval( 'a' ); 
  10.  
  11. ?> 

以上例程的輸出類似于:

  1. a: (refcount=1, is_ref=0)=array ( 
  2.  
  3. 'life' => (refcount=1, is_ref=0)='life' 
  4.  

現(xiàn)在,當(dāng)我們添加一個數(shù)組本身作為這個數(shù)組的元素時,事情就變得有趣,下個例子將說明這個。例中我們加入了引用操作符,否則php將生成一個復(fù)制。

例8 把數(shù)組作為一個元素添加到自己

  1. <?php  
  2.  
  3. $a = array( 'one' ); 
  4.  
  5. $a[] =&$a; 
  6.  
  7. xdebug_debug_zval( 'a' ); 
  8.  
  9. ?> 

以上例程的輸出類似于:

  1. a: (refcount=2, is_ref=1)=array ( 
  2.  
  3. 0 => (refcount=1, is_ref=0)='one'
  4.  
  5. 1 => (refcount=2, is_ref=1)=... 
  6.  

圖示:

 

自引用(curcular reference,自己是自己的一個元素)的數(shù)組的zval

能看到數(shù)組變量 (a) 同時也是這個數(shù)組的第二個元素(1) 指向的變量容器中“refcount”為 2。上面的輸出結(jié)果中的"..."說明發(fā)生了遞歸操作, 顯然在這種情況下意味著"..."指向原始數(shù)組。

跟剛剛一樣,對一個變量調(diào)用unset,將刪除這個符號,且它指向的變量容器中的引用次數(shù)也減1。所以,如果我們在執(zhí)行完上面的代碼后,對變量$a調(diào)用unset, 那么變量 $a 和數(shù)組元素 "1" 所指向的變量容器的引用次數(shù)減1, 從"2"變成"1". 下例可以說明:

例9 Unsetting $a

  1. (refcount=1, is_ref=1)=array ( 
  2.    0 => (refcount=1, is_ref=0)='one'
  3.    1 => (refcount=1, is_ref=1)=... 

圖示:

 

Zvals after removal of array with a circular reference demonstrating the memory leak

清理變量容器的問題(Cleanup Problems)

盡管不再有某個作用域中的任何符號指向這個結(jié)構(gòu)(就是變量容器),由于數(shù)組元素“1”仍然指向數(shù)組本身,所以這個容器不能被清除 。因為沒有另外的符號指向它,用戶沒有辦法清除這個結(jié)構(gòu),結(jié)果就會導(dǎo)致內(nèi)存泄漏。慶幸的是,php將在腳本執(zhí)行結(jié)束時清除這個數(shù)據(jù)結(jié)構(gòu),但是在php清除之前,將耗費不少內(nèi)存。如果你要實現(xiàn)分析算法,或者要做其他像一個子元素指向它的父元素這樣的事情,這種情況就會經(jīng)常發(fā)生。當(dāng)然,同樣的情況也會發(fā)生在對象上,實際上對象更有可能出現(xiàn)這種情況,因為對象總是隱式的被引用。

如果上面的情況發(fā)生僅僅一兩次倒沒什么,但是如果出現(xiàn)幾千次,甚至幾十萬次的內(nèi)存泄漏,這顯然是個大問題。這樣的問題往往發(fā)生在長時間運行的腳本中,比如請求基本上不會結(jié)束的守護進程(deamons)或者單元測試中的大的套件(sets)中。后者的例子:在給巨大的eZ(一個知名的PHP Library) 組件庫的模板組件做單元測試時,就可能會出現(xiàn)問題。有時測試可能需要耗用2GB的內(nèi)存,而測試服務(wù)器很可能沒有這么大的內(nèi)存。 

責(zé)任編輯:龐桂玉 來源: PHP技術(shù)大全
相關(guān)推薦

2023-03-26 22:48:46

Python引用計數(shù)內(nèi)存

2017-03-03 09:26:48

PHP垃圾回收機制

2009-12-09 17:28:34

PHP垃圾回收機制

2017-08-17 15:40:08

大數(shù)據(jù)Python垃圾回收機制

2010-09-25 15:33:19

JVM垃圾回收

2011-07-04 16:48:56

JAVA垃圾回收機制GC

2009-06-23 14:15:00

Java垃圾回收

2017-06-12 17:38:32

Python垃圾回收引用

2021-11-05 15:23:20

JVM回收算法

2021-05-27 21:47:12

Python垃圾回收

2010-09-25 15:26:12

JVM垃圾回收

2010-09-16 15:10:24

JVM垃圾回收機制

2021-12-07 08:01:33

Javascript 垃圾回收機制前端

2010-10-13 10:24:38

垃圾回收機制JVMJava

2015-06-04 09:38:39

Java垃圾回收機

2011-06-28 12:39:34

Java垃圾回收

2020-02-09 17:23:17

Python數(shù)據(jù)字典

2011-07-07 09:54:01

Cocoa Core Foundation

2024-10-28 13:18:54

2011-06-28 10:19:40

C#開發(fā)
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 国产日韩欧美精品 | 日本不卡一二三 | 亚洲激精日韩激精欧美精品 | www.99re| 久久av综合 | 久久精品国产99国产精品 | 久久人人网 | 亚洲小视频 | 久久人人爽人人爽 | www国产成人免费观看视频,深夜成人网 | 国产乱码精品一区二区三区中文 | 91麻豆蜜桃一区二区三区 | 亚洲免费视频一区 | 99热最新| 精品国产高清一区二区三区 | 奇米影视在线 | 久久成人久久 | 免费黄色大片 | 国产良家自拍 | 久久鲁视频 | 精品久久香蕉国产线看观看亚洲 | 精品久久香蕉国产线看观看亚洲 | 亚洲国产高清免费 | 国产美女一区二区 | 久久久久久久国产 | 一区二区在线不卡 | 久久狼人天堂 | 亚洲成人精品在线观看 | 美女久久 | 亚洲乱码国产乱码精品精98午夜 | 日韩av在线一区二区 | 久久久久国产精品免费免费搜索 | 亚洲视频在线免费 | 在线观看免费av网 | 久久大全 | 亚洲韩国精品 | 国产资源在线播放 | 91porn在线观看| 成人不卡一区二区 | 精品中文字幕视频 | 欧美一区二区三区国产精品 |