用cpu接近100%時,你如何找到導致cpu飆升的原因?我的思路是,首先找到進程正在執行的代碼行,從而確定可能有問題的代碼段。然后,再仔細分析有問題的代碼段,從而找出原因。
如果你的程序使用的是c、c++編寫,那么你可以很容易的找到正在執行的代碼行。但是,程序是php編寫的,如何找到可能有問題的代碼行呢?這個問題就是本文要解決的問題。
背景知識:
大家都知道php是一個解釋性語言。用戶編寫的php代碼會生成opcode,由解釋器引擎去解釋執行。在解釋執行過程中,有一個全局變量包含了執行過 程中用到的各種數據。它就是executor_globals。在源碼的Zend/zend_globals.h 文件中可以找到他的類型定義。
這里我們只說兩個對我們比較重要的變量,active_op_array 和 current_execute_data。
active_op_array變量中保存了引擎正在執行的op_array(想了解什么是op_array請點擊查看)。在Zend/zend_compile.h中有關于op_array的數據類型的定義。
看完定義,就不用我多說了把。定義中,filename和 function_name分別保存了正在執行的文件名和方法名。
current_execute_data保存了正在執行的op_array的execute_data。execute_data保存了每個op_array執行過程中的一些數據。其定義在,Zend/zend_compile.h:
定義中的opline就是正在執行的opcode。opcode的結構定義如下:
其中lineno就是opcode所對應的行號。
示例說明:
看完上面的數據結構定義,你是否已經知道如何找php正在執行的文件名,方法名和行號呢?如果還有疑問的話,那就接著看下面的例子。創建一個文件test.php,代碼如下:
cli方式執行php腳本,加入執行的進程號為14973。我們使用gdb命令來調試進程。
很顯然,他正在執行第四行的sleep方法。
如果上面的方法你感覺麻煩,那你可以使用.gdbinit文件。這個文件在php源碼的根目錄下。使用方法如下:
題外話:
從php5.6開始,php中集成了一個phpdbg的工具。可以像gdb調試c語言程序一樣,調試php程序。感興趣的話,可以打開下面的連接看看
新聞熱點
疑難解答