做一個懶惰的PHP開發者
1.調試函數
PHP程序的調試一直是一件讓人頭疼的事,它既不像VB等高級語言那樣有集成的編譯調試環境,也不想Perl那樣可以在Linux或者DOS環境下直接運行。其實,我們完全可以通過靈活地使用echo語句來完成對PHP的調試工作。下面的幾個函數可以讓你隨時查看程序中任何變量的類型及其值。
- function ss_array_as_string (&$array, $column = 0) {
- $str = “Array(n”;
- while(list($var, $val) = each($array)){
- for ($i = 0; $i < $column+1; $i++){
- $str .= “&nbsp;&nbsp;&nbsp;&nbsp;”;
- }
- $str .= $var. ==> ;
- $str .= ss_as_string($val, $column+1).” n”;
- }
- for ($i = 0; $i < $column; $i++){
- $str .= “&nbsp;&nbsp;&nbsp;&nbsp;”;
- }
- return $str.);
- }
- function ss_object_as_string (&$object, $column = 0) {
- if (empty($object->classname)) {
- return “$object”;
- }
- else {
- $str = $object->classname.”( n”;
- while (list(,$var) = each($object->persistent_slots)) {
- for ($i = 0; $i < $column; $i++){
- $str .= “&nbsp;&nbsp;&nbsp;&nbsp;”;
- }
- global $$var;
- $str .= $var. ==> ;
- $str .= ss_as_string($$var, column+1).” n”;
- }
- for ($i = 0; $i < $column; $i++){
- $str .= “&nbsp;&nbsp;&nbsp;&nbsp;”;
- }
- return $str.);
- }
- }
- function ss_as_string (&$thing, $column = 0) {
- if (is_object($thing)) {
- return ss_object_as_string($thing, $column);
- }
- elseif (is_array($thing)) {
- return ss_array_as_string($thing, $column);
- }
- elseif (is_double($thing)) {
- return “Double(“.$thing.”)”;
- }
- elseif (is_long($thing)) {
- return “Long(“.$thing.”)”;
- }
- elseif (is_string($thing)) {
- return “String(“.$thing.”)”;
- }
- else {
- return “Unknown(“.$thing.”)”;
- }
- }
需要的時候,在程序中簡單地加入下面的一條代碼即可查看程序中的所使用的變量(包括數組和對象)的類型和值:
- echo ss_as_string($my_variable);
使用下面的語句,我們可以直接查看程序中所有的變量的值:
- echo ss_as_string($GLOBALS);
3. 控制Log信息的函數
調試PHP程序的另外一種重要的方法就是查看Log信息。如果能夠方便地控制Log信息的級別以及Log信息的顯示內容,將會給程序調試帶來更多的便利。下面的幾個函數可以方便地實現這個功能。
- $ss_log_level = 0;
- $ss_log_filename = /tmp/ss-log;
- $ss_log_levels = array(
- NONE => 0,
- ERROR => 1,
- INFO => 2,
- DEBUG => 3);
- function ss_log_set_level ($level = ERROR) {
- global $ss_log_level;
- $ss_log_level = $level;
- }
- function ss_log ($level, $message) {
- global $ss_log_level, $ss-log-filename;
- if ($ss_log_levels[$ss_log_level] < $ss_log_levels[$level]) {
- // 不顯示Log信息
- return false;
- }
- $fd = fopen($ss_log_filename, “a+”);
- fputs($fd, $level. - [.ss_timestamp_pretty().] - .$message.”n”);
- fclose($fd);
- return true;
- }
- function ss_log_reset () {
- global $ss_log_filename;
- @unlink($ss_log_filename);
- }
在上面的函數中,有四個Log級別變量。運行PHP程序時,只有當Log的級別低於預設的級別值時,Log信息纔可以被記錄和顯示出來。例如,在程序中加入如下的一條語句:
- ss_log_set_level(INFO);
那麼,運行PHP程序時,只有ERROR和INFO級別的LOG信息才能被記錄和顯示出來,DEBUG級的信息則被忽略了。除此之外,我們還可以設定顯示的信息內容,其語句如下:
- ss_log(ERROR, “testing level ERROR”);
- ss_log(INFO, “testing level INFO”);
- ss_log(DEBUG, “testing level DEBUG”);
你也可以隨時使用下面的語句清空LOG信息:
- ss_log_reset();
4.速度測試函數
爲了優化代碼,我們需要一種可以測試代碼運行時間的方法,從而來選擇最優的代碼。下面的函數可以測試運行代碼所需的時間:
- function ss_timing_start ($name = default) {
- global $ss_timing_start_times;
- $ss_timing_start_times[$name] = explode( , microtime());
- }
- function ss_timing_stop ($name = default) {
- global $ss_timing_stop_times;
- $ss_timing_stop_times[$name] = explode(, microtime());
- }
- function ss_timing_current ($name = default) {
- global $ss_timing_start_times, $ss_timing_stop_times;
- if (!isset($ss_timing_start_times[$name])) {
- return 0;
- }
- if (!isset($ss_timing_stop_times[$name])) {
- $stop_time = explode(, microtime());
- }
- else {
- $stop_time = $ss_timing_stop_times[$name];
- }
- $current = $stop_time[1] - $ss_timing_start_times[$name][1];
- $current += $stop_time[0] - $ss_timing_start_times[$name][0];
- return $current;
- }
現在可以輕鬆地檢查任何一段代碼的執行時間了,甚至我們可以同時使用多個計時器,只需在使用上述的幾個函數時設定不同的參數作爲計時器的名稱就可以了。
5.調試和優化數據庫的操作
對於數據庫來說,運行速度是至關重要的。儘管很多書籍和文章都講授了一些快速運行數據庫的方法,但是所有的方法都必須經過實踐的檢驗。下面我們將把PHPLib函數庫中的query()函數和上面介紹的幾個函數綜合起來編寫成新的query()函數,和原先的函數相比,這個函數增加了運行時間的監測功能。
- function query($Query_String, $halt_on_error = 1) {
- $this->connect();
- ss_timing_start();
- $this->Query_ID = @mysql_query($Query_String,$this->Link_ID);
- ss_timing_stop();
- ss_log(INFO, ss_timing_current(). Secs - .$Query_String);
- $this->Row = 0;
- $this->Errno = mysql_errno();
- $this->Error = mysql_error();
- if ($halt_on_error && !$this->Query_ID) {
- $this->halt(“Invalid SQL: “.$Query_String);
- }
- return $this->Query_ID;
- }