熱點推薦:
您现在的位置: 電腦知識網 >> 編程 >> PHP編程 >> 正文

如何成為PHP高手

2013-11-15 12:25:46  來源: PHP編程 

  PHP是一門高效的網絡編程語言由於它具有編寫靈活運行快速等優點迅速成為Web程序員的首選語言那麼怎樣才能成為一個優秀的PHP開發者呢?

  要成為一名PHP編程高手並不容易並不像很多人想象的那樣只要能夠飛快地編寫幾條簡單的代碼去解決一個復雜的問題就是PHP編程高手了真正的PHP高手還需要考慮更多的其它問題以下三條准則是一名成熟的PHP程序員在編程中應該首先遵循的准則

  ◆懶惰是金

  ◆編寫漂亮的代碼

  ◆追求程序的速度而不是編程的速度

  懶惰是金

  做一個懶惰的程序員嗎?這個想法太奇怪了!因為這個世界上最忙碌的人可能就是計算機程序員了但正是因為程序員太忙了所以才應該在編程時學會偷懶對於一個程序員來說懶惰的方法有兩種

  其一大膽使用現成的別人的程序代碼把這些代碼融入到你自己的程序或者項目中去其二是編寫一些有用的代碼建立一個函數庫在將來編寫程序時可以順手拈來省去了許多重復的勞動自然就可以懶惰一點了這兩種偷懶的方法都非常適合PHP程序員了

  首先PHP是在自由開放的環境中誕生和成長的一門語言在世界各地有成千上萬的程序員他們一直在為PHP的完美而不斷奮斗他們也願意和別人分享自己的聰明才智和自己編寫的代碼你每天都可以從一些PHP網站郵件列表新聞組發現大量的優秀的程序代碼

  這樣說我並不是鼓勵你整天等著讓別人為你編寫代碼但是你可以“站在偉人的肩膀上”充分發揚“拿來主義”聰明地應用別人的程序代碼可以節省你大量時間其次在PHP中你可以方便地建立自己的函數庫這樣可以在你以後編寫程序時省去很多麻煩

  下面筆者為大家介紹幾個通用的函數這些函數有的來自網上的一些開放源代碼的項目有的精選自郵件列表如果你能把它們加入到你自己的函數庫中遲早你將會發現自己受益無窮

  通用數據庫處理函數

  和其它的CGI函數相比PHP的優點之一是具有很強大的數據庫處理能力但是在PHP中對於不同的數據庫都使用一些特定的函數來專門處理缺少通用的數據庫處理函數這大大降低了程序代碼的可移植性這也為初學編程的朋友帶來了很多不便

  

  在網上許多程序員都通過封裝類解決了這個問題他們編寫了統一的函數用來處理任何流行的數據庫——不管是在Linux世界深受歡迎的Mysql還是在Windows平台上廣泛流行的SqlServer

  就筆者個人來說非常喜歡使用這些函數因為可以直接使用一些簡單的諸如”query””next_record”之類的函數而不需要考慮數據庫的連接數據庫句柄這些復雜的東西更不需要考慮使用的是何種數據庫如果你需要這些函數你可以通過訪問以下的幾個網址而得到

  ◆

  ◆/package/

  ◆

  變量調試函數

  PHP程序的調試一直是一件讓人頭疼的事它既不像VB等高級語言那樣有集成的編譯調試環境也不想Perl那樣可以在Linux或者DOS環境下直接運行其實我們完全可以通過靈活地使用echo語句來完成對PHP的調試工作下面的幾個函數可以讓你隨時查看程序中任何變量的類型及其值

  function ss_array_as_string (&$array $column = ) {  $str = "Array(n"; while(list($var $val) = each($array)){  for ($i = ; $i < $column+; $i++){ $str = " ";  }  $str = $var ==> ;  $str = ss_as_string($val $column+)" n"; } for ($i = ; $i < $column; $i++){  $str = " "; } return $str);  }  function ss_object_as_string (&$object $column = ) { if (emptyempty($object>classname)) {  return "$object"; } else {  $str = $object>classname"( n"; while (list($var) = each($object>persistent_slots)) {  for ($i = ; $i < $column; $i++){ $str = " ";  }  global $$var;  $str = $var ==> ;  $str = ss_as_string($$var column+)" n"; } for ($i = ; $i < $column; $i++){  $str = " "; } return $str); }  }  function ss_as_string (&$thing $column = ) {  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);

   控制Log信息的函數

  調試PHP  程序的另外一種重要的方法就是查看Log信息如果能夠方便地控制Log信息的級別以及Log信息的顯示內容將會給程序調試帶來更多的便利下面的幾個函數可以方便地實現這個功能

  $ss_log_level = ; $ss_log_filename = /tmp/sslog; $ss_log_levels = array(  NONE =>  ERROR =>  INFO =>  DEBUG => ); function ss_log_set_level ($level = ERROR) {  global $ss_log_level;  $ss_log_level = $level; } function ss_log ($level $message) {  global $ss_log_level $sslogfilename;  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信息

  速度測試函數

  為了優化代碼我們需要一種可以測試代碼運行時間的方法從而來選擇最優的代碼下面的函數可以測試運行代碼所需的時間

  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 ;  }  if (!isset($ss_timing_stop_times[$name])) { $stop_time = explode( microtime());  }  else { $stop_time = $ss_timing_stop_times[$name];  }  $current = $stop_time[] $ss_timing_start_times[$name][];  $current += $stop_time[] $ss_timing_start_times[$name][];  return $current; }

  現在可以輕松地檢查任何一段代碼的執行時間了甚至我們可以同時使用多個計時器只需在使用上述的幾個函數時設定不同的參數作為計時器的名稱就可以了

  調試和優化數據庫的操作

  對於數據庫來說運行速度是至關重要的盡管很多書籍和文章都講授了一些快速運行數據庫的方法但是所有的方法都必須經過實踐的檢驗下面我們將把PHPLib函數庫中的query()函數和上面介紹的幾個函數綜合起來編寫成新的query()函數和原先的函數相比這個函數增加了運行時間的監測功能

  

  function query($Query_String $halt_on_error = ) {  $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 = ;  $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; } 編寫漂亮的代碼

  將後台程序與前端程序分開

  在編寫PHP程序時有些代碼是用來處理一些事務例如操作數據庫進行數學運算等而另外的一些代碼則只是事務處理的結果顯示出來例如一些使用echo語句將結果以HTML的格式顯示在Web浏覽器上的PHP代碼以及那些直接嵌入PHP程序的HTML代碼首先我們應該清晰地區分這兩種代碼把前者稱為後台程序把後者稱為前端程序

  因為PHP是一種嵌入式編程語言也就是說所有的PHP代碼都可以嵌入到HTML代碼之中這為程序的編寫帶來了許多便利之處但是“物極必反”如果在一段較長的程序中將PHP代碼和HTML代碼混合編寫這將使程序雜亂無章不利於程序的維護和閱讀

  所以我們需要盡可能地將這些程序中混雜於HTML代碼中的PHP代碼移植出來在專門的文件中將這些代碼封裝成函數然後在HTML代碼中使用include語句來包含這些文件在適當的位置調用這些函數即可

  這種做法一方面使HTML代碼和PHP代碼都簡單易讀另一方面因為HTML代碼需要不斷更新而這種分離的方法可以確保後台程序不會被破壞同前端程序不同後台程序更多追求的是穩定結構化極少更改所以應該認真地設計和管理其實在設計台程序時投入大量時間是值得的“現在栽樹以後乘涼”在以後的設計工作中將可以輕松地使用現在編寫的後台程序

  靈活使用包含文件

  正如前面所說的那樣後台程序應當安排在一系列的包含文件中包含文件可以通過include語句在需要時動態裝入也可以在phpini文件中通過使用auto_prepend_file指令預先自動裝入如果使用後一種方法的話雖然取得了一勞永逸的好處但是也有一些缺點值得我們注意下面的一段代碼向我們展示了解析一個龐大的包含文件需要一定的時間

  require(timinginc);

  ss_timing_start();

  include(testinc);

  ss_timing_stop();

  echo

  ss_timing_current()

  ;

  在上面的代碼中testinc是一個行的包含文件運行的結果顯示解析這個包含文件花費了秒鐘對於一個大型網站來說這個速度並不是可以忽略不記的使用包含文件的另外一個缺點是如果一個文件中的一個語句發生錯誤將會使整個網站的PHP  程序都無法運行所以使用起來也及其小心其實對包含文件稍做處理即可以使包含文件只在需要時進行解析下面的代碼使abcinc文件只在程序需要時才作解析

  if ( defined( __LIBA_INC) ) return;

  define( __LIBA_INC );

  /* * 代碼 */

  使用面向對象的編程方法

  PHP也是一種面向對象的語言面向對象的編程方法是優秀的程序員們非常推崇的一種軟件設計方法在PHP編程中可以充分發揮面向對象語言的優勢對編程中的對象進行封裝在前面的代碼中我們使用了面向對象的方法例如在管理數據庫時我們將query()函數封裝進數據庫類中這極大地方便了代碼的管理增加了程序的可讀性

  追求程序速度而不是編程的速度

  在網站建設中程序運行速度和網頁下載速度都是關系成敗的重要因素作為一名Web程序員應該更加注意代碼的運行速度下面介紹的幾種方法都在不同程度上提高了代碼的運行速度

  使用內嵌的HTML代碼而不是PHP的echo語句

  因為PHP是一門嵌入式Web編程語言可以將HTML代碼和PHP代碼相互嵌入但是很多程序員擔心在HTML代碼中過多的使用""嵌入PHP代碼會多次調用PHP解釋器從而降低了PHP代碼的運行速度所以寧願使用PHP的echo語句來輸出HTML代碼而不直接使用HTML代碼

  但事實卻恰恰相反每一個PHP頁面只調用一次PHP解釋器來解釋所有的PHP代碼所以只在需要時才嵌入PHP代碼而大多數的時候直接使用HTML代碼輸入結果不但不會降低程序的運行速度而且因為減少了對echo語句的解析往往可以提高代碼的運行速度下面的一段代碼證明了我們的結論在這段代碼中我們使用了前面介紹的時間測試函數

  使用strreplace而不是eregreplace

  習慣使用Perl進行編程的程序員更加願意使用ereg_replace完成字符串替換工作因為在PHP中ereg_replace的用法和Perl中模式匹配的用法相近但是下面的這段代碼證明使用str_replace 代替 ereg_replace將可以大大提高代碼的運行速度測試str_replace和ereg_replace的運行速度

  //這段代碼測試str_replace的運行速度 emphasis; ?>

  

  for ($i=; $i<; $i++) {

  str_replace(i> b> $string)

  ;

  }

  //這段代碼測試ereg_replace的運行速度

  for ($i=; $i<; $i++) {

  ereg_replace(<([/]*)i> <b> $string)

  ;

  }

  注意字符串的引用

  PHP和其它很多編程語言一樣可以使用雙引號("")來引用字符串也可以使用單引號()但是在PHP中如果使用雙引號來引用字符串那麼PHP解析器將首先分析字符串中有沒有對變量的引用有變量的話將對變量進行替換如果是單引號則沒有如此復雜——直接將單引號包含起來的所有字符串直接顯示出來顯然在PHP編程中如果使用單引號引用字符串變量要比使用雙引號快速一些

  在數據庫中避免使用聯合操作

  比起其它的Web編程語言來說PHP的數據庫功能十分強大但是在PHP中數據庫的運行仍然是一件十分費時費力的事情所以作為一個Web程序員要盡量減少數據庫的查詢操作同時應該為數據庫建立適當的索引

  另一件值得注意的事情是在用PHP操作數據庫時盡可能不使用多個數據表的聯合操作盡管聯合操作可以增強數據庫的查詢功能但是卻大大增加了服務器的負擔為了說明這個問題我們可以看看下面的這個簡單的例子

  我們在數據庫中創建了兩個數據表foo和big_foo在數據表foo中只有一個字段包含了從之間的所有自然數數據表big_foo同樣只有一個字段但包含了從之間的全部自然數所以從大小上說big_foo等於foo與它自身進行了聯合操作

  $db>query("select * from foo");

   secs

  $db>next_record();

   secs

  $db>query("insert into foo values (NULL)");

   secs

  $db>query("select * from foo as a foo as b");

   secs

  $db>query("select * from foo as a foo as b where aid > bid");

   secs

  $db>query("select * from foo as a foo as b where aid = bid");

   secs

  $db>query("select * from big_foo");

   secs

  從上面操作結果我們可以發現對於兩個有條記錄的數據表進行聯合其速度並不比對一個條紀錄的大型數據表單獨進行操作快多少

  注意include與require的區別

  

  在PHP變成中include()與require()的功能相同但在用法上卻有一些不同include()是有條件包含函數而require()則是無條件包含函數例如在下面的一個例子中如果變量$somgthing為真則將包含文件somefile

  if($something){

  include("somefile");

  }

  但不管$something取何值下面的代碼將把文件somefile包含進文件裡

  if($something){

  require("somefile");

  }

  下面的這個有趣的例子充分說明了這兩個函數之間的不同

  $i = ;

  while ($i < ) {

  require("somefile$i");

  $i++;

  }

  在這段代碼中每一次循環的時候程序都將把同一個文件包含進去很顯然這不是程序員的初衷從代碼中我們可以看出這段代碼希望在每次循環時將不同的文件包含進來如果要完成這個功能必須求助函數include();

  $i = ;

  while ($i < ) {

  include("somefile$i");

  $i++;

  }

  注意echo和print的區別

  PHP中echo和print的功能也基本相同但是兩者之間也有細微差別在PHP代碼中可以把print作為一個普通函數來使用例如執行下面的代碼後變量$res的值將為

  $ret = print "Hello World";

  這意味著print可用在一些復雜的表達式中而echo則不行同樣在代碼中echo語句的運行速度要略微快於print語句因為echo語句不要求返回任何數值


From:http://tw.wingwit.com/Article/program/PHP/201311/20928.html
    推薦文章
    Copyright © 2005-2013 電腦知識網 Computer Knowledge   All rights reserved.