php內存回
Ⅰ 有人研究過php的內存回收機制嗎
表示不敢研究...
php垃圾回收機制GC,每個版本都在更新...你研究他幹嘛!直接研究內存原理就行了....
php.ini的內存主要是防止,php佔用超過系統內存,造成死機...
配置文件里的內存只是單次解釋佔用的內存....
cup和內存的消耗的關系有一個臨界點....就是最優值,
只想效率就是假設內存無限大,io帶寬無限大,cpu緩存無限大的情況,cpu佔用無限小....
Ⅱ 如何獲知PHP程序佔用多少內存memory
PHP性能優化過程中需要獲取PHP內存消耗,使用memory_get_usage()函數可獲取當前的內存消耗情況,函數使用簡單,這里討論一下memory_get_usage()函數的用法與實例
三,基礎用法與實例
1,獲取當前的內存消耗量
復制代碼代碼如下:
<?php
echo memory_get_usage();
$var = str_repeat("liuhui", 10000);
echo memory_get_usage();
unset($var);
echo memory_get_usage();
?>
分別輸出:62328 122504 62416
說明:memory_get_usage()函數輸出的數值為bytes單位
2,格式化memory_get_usage()輸出
復制代碼代碼如下:
<?php
function convert($size){
$unit=array('b','kb','mb','gb','tb','pb');
return @round($size/pow(1024,($i=floor(log($size,1024)))),2).' '.$unit[$i];
}
echo convert(memory_get_usage(true));
?>
輸出:256 kb
3,自定義函數獲取數組或變數值大小
復制代碼代碼如下:
<?php
function array_size($arr) {
ob_start();
print_r($arr);
$mem = ob_get_contents();
ob_end_clean();
$mem = preg_replace("/\n +/", "", $mem);
$mem = strlen($mem);
return $mem;
}
$memEstimate = array_size($GLOBALS);
?>
Ⅲ php內存分配怎麼理解
1、內存檢查,如果太大則報錯
2、如果命中緩存,則使用fastcache中內存塊
3、在zendMM中heap層根據申請大小從小塊內存、大塊內存、剩餘內存中查找合適內存
4、如果還沒合適的,則直接再申請一塊內存
5、重設heap狀態
6、返回內存地址、標志變數維護
Ⅳ php 內存不夠怎麼解決
這個提示的意思是內存條和磁碟空間太小,系統分配臨時緩存的空間也隨之不足,使你在運行一些程序時受到了緩存的限制。比方你在打游戲時是一關一關的往前打,你的電腦就要把你打過的游戲暫時保存在這個虛擬內存里。如果空間存儲已滿,那麼你正在運行中的程序就會受到一些限制。
要解決虛擬內存太低的問題,只有把內存條和磁碟的容量擴大,一般內存條要達到1gb以上,磁碟也要達到80g的才好用。建議你只花幾十元加一條內存條問題就全解決了。
如果你要手動將虛擬內存的緩存空間指定到其它盤符上是沒有多大效果的。因為造成此問題的主要原因就是你的這些空間太小,使你的系統在合理分配時受到了限制。所以,你使用手動強迫手段來調配完全是錯誤的,相反,這樣對你的電腦或更有所不利。
如果
你的確要手動調整,方法是:滑鼠指向「我的電腦」彈右鍵——單擊「屬性」——單擊「高級」——單擊「性能」這里的「設置」——單擊「高級」——單擊「虛擬內存」這里的「更改」——在上面的列表中選擇你要將虛擬內存更改到哪個盤,(選擇剩餘空間比較大的磁碟,一般的是e盤)——單擊「自定義大小
」——輸入最大值和最小值(把最大值改為物理內存的2.5倍)——點擊右下角的「設置」按鈕
——這時會彈出「重新啟動計算機」的提示——單擊「確定」重啟即可
Ⅳ 查看PHP程序佔用多少內存
想要知道編寫的 PHP 腳本需要佔用多少內存么?很簡單,直接使用 PHP 查看當前分配給 PHP 腳本的內存的函數 memory_get_usage() 就可以了$memory = round(memory_get_usage() / 1024 / 1024, 2).'MB';
上面的程序默認輸出單位為 byte(s))我將它轉換成MB,也就是當時 PHP 腳本使用的內存(不含 memory_get_usage() 函數本身佔用的內存)。
更多問題可以去php中文網問答社區提問http://www.php.cn/wenda.html,大神在線幫你解決,希望對你有幫助
Ⅵ PHP處理內存問題,怎麼解決
最簡單或常用的方法是修改php.ini
首先找到對你的網站生效的php.ini文件
由於有多個地方都可以設置php的參數,找到正確的配置文件,並進行更改是首先要做的一步。如果你上面的方法建立了php文件來查看其配置參數,則你可以找到"Loaded Configuration File"這一項,以下是個例子:
對於Linux用戶,你可以通過執行"php -i | grep Loaded Configuration File"來找到對應的配置文件。而Windows用戶,你可以嘗試修改你的php安裝目錄下的php.ini。
編輯php.ini
在php.ini中,找到"memory_limit"這一項,如果沒有,你可以在文件的尾部自己增加這個參數。以下是一些設置範例
memory_limit = 128M ; 可以將128M改為任何你想設置的值保存文件
重啟web 伺服器
如果是web伺服器使用Apache, 則執行:
httpd restart
有些情況下,你可能不被允許私修改php.ini。比如如果你購買了虛擬主機服務,但是你的服務商確禁止你修改這個文件。那麼,你可以需要考慮用其他方法來增加memory_limit的值。
Ⅶ php內存不足
你好
時候我們再運行php程序時,會出現
Allowed memory size of 8388608 bytes exhausted (tried to allocate 1298358 bytes)
出現該錯誤的原因:在確保不是程序產生的原因(例如死循環),是由於php頁面消耗的最大內存默認是為 8M (在PHP的ini件里可以看到) ,
如果文件太大 或圖片太大 在讀取的時候 會發生上述錯誤。
解決辦法:
1.修改 php.ini將memory_limit由 8M 改成 16M(或更大),重啟apache服務
2.在PHP 文件中 加入 ini_set(『memory_limit',『**M');注意:為了系統的其它資源的正常使用 請您不要將 memory_limit設置太大,其中-1為不限
3.修改.htaccess 文檔(前提是該目錄支持.htaccess)在文檔中新增一句:php_value memory_limit 16M(或更大)
滿意請採納
Ⅷ php 難題 內存佔用
兩個概念不同,函數獲取是PHP.EXE(或者你的HTTPD.EXE)給此腳本分配的內存,而操作系統分配給進程的內存會大很多,我下面的沒有使用任何變數的簡單語句腳本,就兩個語句,一個輸出內存使用,一個暫停一下,程序顯示腳本分陪內存100多K,而操作系統顯示3M,如下圖:
當然,你可以看看你的簡單腳本佔用內存多少,上G確實太驚人了。
Ⅸ PHP命令執行PHP腳本,結束之前,內存會回收嗎
再詳細說下問題:
unix下,用php命令來執行php腳本,在php結束之前,內存有沒有機會被回收?新的GC演算法有沒有機會被調用?
出現這個問題,是因為線上有個離線數據導入腳本,需要把幾千萬行數據篩選入庫,發現,在執行過程中,到達一定程度,就會拋出內存使用超過最大值。
1Fatalerror:
那第一想到的就是程序是不是有什麼bug,造成內存超出,看了半天沒有發現問題,於是,突然出現了這個疑問。那要解決這個疑問,最好的辦法就去翻源碼吧。
在之前我這么說:
都知道,PHP5.3有了新的垃圾回收機制:GC,它不是重點,不談它的原理。
經過翻閱PHP源碼,發現,調用這個的時機是在main/main.c::php_request_shutdown這個函數中,
12/*7.Shutdownscanner/executor/compilerandrestoreinientries*/zend_deactivate(TSRMLS_C);
php_request_shutdown,通過名字就能看出,它是在php請求結束的時候執行的,在這里會執行gc_collect_cycles來清理內存。
其實這句話是沒錯,但它只針對於SAPI介面(之前我就錯在這個地方。),在用PHP命令執行php腳本的時候,是不會執行這個php_request_shutdown的。
那回到最初的問題,過程中到底有沒有執行GC呢?
為了更直觀有效的知道答案,我選擇了最BT,最暴力的方法,攔截gc_collect_cycles,輸出error_log到文件,只要執行了,
那肯定會輸出log來。
重新編譯PHP後,果不其然,符合官方的說法,只要buckets滿超過默認值1000,就會啟動GC來清理沒用的內存,防止內存泄露。
那問「什麼時間觸發的GC呢?」,答「buckets超過1000的時候啊」,這不屁話嘛,要的是真真正正的執行流程,so。。不斷的debug,
不斷的grep,不斷的step,不斷的C+T,終於搞清楚了。下面就來根據官方的說法詳細談談,PHP到底是怎麼觸發的。
有一點要注意,PHP的命令入口和sapi介面的入口是不同的,我就載在這個地方,以為都公用一個。
測試代碼以官方文檔為例:
1234567891011121314<?phpclassFoo{public$var='3.1415962654';}for($i=0;$i<=1000000;$i++){$a=newFoo;$a->self=$a;}echomemory_get_peak_usage()," ";?>
這樣的代碼,在PHP5.3之前,肯定會造成大量的內存泄露,不過,誰在開發時又能開發出這么變態的代碼來?除非這個人很變態。^.*
那PHP的命令入口是什麼?流程又是什麼?
主要函數流程如下:
入口main函數(sapi/cli/php_cli.c)==》php_execute_script(main/main.c)==>zend_execute_scripts(Zend/zend.c)==>execute(Zend/zend_vm_execute.h)
調用GC的地方在execute里。
簡單描述下這個過程,
main是入口,它的作用是根據我們傳遞的參數做不同的設置,最後會把我們的php腳本作為一個zend_file_handle指針傳遞給
php_execute_script函數,zend_file_handle其實就是把FILE*做了一下封裝,保存了一些其他的文件信息。
php_execute_script會做一些文件檢查工作,把php腳本加到哈希表included_files中。
php_execute_scripts會執行zend_compile_file函數來解釋我們寫的PHP代碼,最後執行execute。
應該都知道Zend把腳本解析完會生成op代碼保存到哈希表:active_op_array中,execute會逐個執行每個op,
op基本上都對應一個ZEND_ASSIGN_*_HANDLER這樣的一個宏,它就保存在active_op_array->opline->handlers中。
在進入到execute之後:
首先初始化execute_data,它保存了很多重要信息,上下文信息,然後調用ZEND_VM_SET_OPCODE宏,
把execute_data->opline的指針指向active_op_array->opline->handlers。
之後,execute會執行一個while循環,逐條執行opline:
(1){intret;#ifdefZEND_WIN32if(EG(timed_out)){zend_timeout(0);}#endifif((ret=EX(opline)->handler(execute_dataTSRMLS_CC))>0){switch(ret){case1:EG(in_execution)=original_in_execution;return;case2:op_array=EG(active_op_array);gotozend_vm_enter;case3:execute_data=EG(current_execute_data);default:break;}}}
每個handlers都會執行一個宏:ZEND_VM_NEXT_OPCODE(),它意思就是跳到下一個Opline,這樣就能逐條執行了。
最後跟蹤上面的PHP代碼會執行ZEND_ASSIGN_SPEC_CV_VAR_HANDLER這個宏,它是幹嘛的?他就是變數賦值
下面代碼執行的操作:
1234classA{}$a=newA();
這里就會執行這個宏。
在這個宏里有段代碼:
_FASTCALLZEND_ASSIGN_SPEC_CV_VAR_HANDLER(ZEND_OPCODE_HANDLER_ARGS){zend_op*opline=EX(opline);zend_free_opfree_op2;zval*value=_get_zval_ptr_var(&opline->op2,EX(Ts),&free_op2TSRMLS_CC);zval**variable_ptr_ptr=_get_zval_ptr_ptr_cv(&opline->op1,EX(Ts),BP_VAR_WTSRMLS_CC);if(IS_CV==IS_VAR&&!variable_ptr_ptr){if(zend_assign_to_string_offset(&EX_T(opline->op1.u.var),value,IS_VARTSRMLS_CC)){if(!RETURN_VALUE_UNUSED(&opline->result)){EX_T(opline->result.u.var).var.ptr_ptr=&EX_T(opline->result.u.var).var.ptr;ALLOC_ZVAL(EX_T(opline->result.u.var).var.ptr);INIT_PZVAL(EX_T(opline->result.u.var).var.ptr);ZVAL_STRINGL(EX_T(opline->result.u.var).var.ptr,Z_STRVAL_P(EX_T(opline->op1.u.var).str_offset.str)+EX_T(opline->op1.u.var).str_offset.offset,1,1);}}elseif(!RETURN_VALUE_UNUSED(&opline->result)){AI_SET_PTR(EX_T(opline->result.u.var).var,EG(uninitialized_zval_ptr));PZVAL_LOCK(EG(uninitialized_zval_ptr));}}else{value=zend_assign_to_variable(variable_ptr_ptr,value,0TSRMLS_CC);if(!RETURN_VALUE_UNUSED(&opline->result)){AI_SET_PTR(EX_T(opline->result.u.var).var,value);PZVAL_LOCK(value);}}/*zend_assign_to_variable()alwaystakescareofop2,neverfreeit!*/if(free_op2.var){zval_ptr_dtor(&free_op2.var);};ZEND_VM_NEXT_OPCODE();}
free_op2.var保存的是newA的對象.
free_op2.var這個是哪兒來的呢?
在整個execute期間,維持一個execute_data結構,裡面有個Ts指針
1union_temp_variable*Ts;
它用來保存一些臨時的變數信息,比如newA(),這個會保存到Ts鏈表裡,
opline->op2.u.var這個裡面保存了此臨時變數所在的位置,然後Ts+這個值是一個zval*指針,它就保存了newA產生的對象.
在代碼中
1if(free_op2.var){zval_ptr_dtor(&free_op2.var);};
zval_ptr_dtor會根據free_op2.var的值執行到Zend/zend_execute_API.c::_zval_ptr_dtor函數中,
_APIvoid_zval_ptr_dtor(zval**zval_ptrZEND_FILE_LINE_DC)/*{{{*/{zval*zv=*zval_ptr;#ifDEBUG_ZEND>=2printf("Recingrefcountfor%x(%x):%d->%d ",*zval_ptr,zval_ptr,Z_REFCOUNT_PP(zval_ptr),Z_REFCOUNT_PP(zval_ptr)-1);#endifZ_DELREF_P(zv);if(Z_REFCOUNT_P(zv)==0){TSRMLS_FETCH();if(zv!=&EG(uninitialized_zval)){GC_REMOVE_ZVAL_FROM_BUFFER(zv);zval_dtor(zv);efree_rel(zv);}}else{TSRMLS_FETCH();if(Z_REFCOUNT_P(zv)==1){Z_UNSET_ISREF_P(zv);}GC_ZVAL_CHECK_POSSIBLE_ROOT(zv);}}
GC_ZVAL_CHECK_POSSIBLE_ROOT(zv);
它就是最終GC演算法執行的地方.
gc_collect_cycles就在這個宏中執行了..
所以..
回到上面的問題,
php無論在SAPI介面或命令端,都會執行GC演算法來進行垃圾內存回收.
Ⅹ php的最大內存限制是多少
這個看情況了,處理大數據肯定需要的內存更大,只是本地調試肯定用不了多少。
php.ini的內存限制默認為memory_limit = 128M,只要把memory_limit設為-1,就不會限制你跑cli腳本時的內存限制。
一般作為一台專門運行php的伺服器,肯定要盡可能的把內存給於php運行,一台不夠就追加。