介紹一些關于改善和優化PHP代碼的提示和技巧。請注意,這些PHP技巧適用于初學者,而不是那些已經在使用MVC框架的人。
1.不要使用相對路徑,要定義一個根路徑
這樣的代碼行很常見:
require_once('../../lib/some_class.php');
這種方法有很多缺點:
它首先搜索php包括路徑中的指定目錄,然后查看當前目錄。因此,會檢查許多目錄。
當一個腳本被包含在另一個腳本的不同目錄中時,它的基本目錄變為包含腳本的目錄。
另一個問題是,當一個腳本從cron運行時,它可能不會將它的父目錄作為工作目錄。
所以使用絕對路徑便成為了一個好方法:
define('ROOT' , '/var/www/project/');
require_once(ROOT . '../../lib/some_class.php');//rest of the code
這就是一個絕對路徑,并且會一直保持不變。但是,我們可以進一步改善。目錄/var/www/project可以變,那么我們每次都要改嗎?
不,使用魔術常量如__FILE__可以讓它變得可移植。請仔細看:
//suppose your script is /var/www/project/index.php//Then __FILE__ will always have that full path.define('ROOT' , pathinfo(__FILE__, PATHINFO_DIRNAME));require_once(ROOT . '../../lib/some_class.php');//rest of the code
所以現在,即使你將項目轉移到一個不同的目錄,例如將其移動到一個在線的服務器上,這些代碼不需要更改就可以運行。
2.不使用require,包括require_once或include_once
你的腳本上可能會包括各種文件,如類庫,實用程序文件和輔助函數等,就像這些:
require_once('lib/Database.php');
require_once('lib/Mail.php');
require_once('helpers/utitlity_functions.php');
這相當粗糙。代碼需要更加靈活。寫好輔助函數可以更容易地包含東西。舉個例子:
function load_class($class_name){ //path to the class file
$path = ROOT . '/lib/' . $class_name . '.php'); require_once( $path );
}
load_class('Database');
load_class('Mail');
看到區別了嗎?很明顯。不需要任何更多的解釋。
你還可以進一步改善:
function load_class($class_name){
//path to the class file
$path = ROOT . '/lib/' . $class_name . '.php');
if(file_exists($path))
{
require_once( $path );
}
}
這樣做可以完成很多事情:
為同一個類文件搜索多個目錄。
輕松更改包含類文件的目錄,而不破壞任何地方的代碼。
使用類似的函數用于加載包含輔助函數、HTML內容等的文件。
3.在應用程序中維護調試環境
在開發過程中,我們echo數據庫查詢,轉儲創造問題的變量,然后一旦問題被解決,我們注釋它們或刪除它們。但讓一切留在原地可提供長效幫助。
在開發計算機上,你可以這樣做:
define('ENVIRONMENT' , 'development');
if(! $db->query( $query )
{
if(ENVIRONMENT == 'development')
{
echo "$query failed";
}
else
{
echo "Database error. Please contact administrator";
}
}
并且在服務器上,你可以這樣做:
define('ENVIRONMENT' , 'production');
if(! $db->query( $query )
{ if(ENVIRONMENT == 'development')
{
echo "$query failed";
} else
{
echo "Database error. Please contact administrator";
}
}
4.通過會話傳播狀態消息
狀態消息是那些執行任務后生成的消息。
<?php
if($wrong_username || $wrong_password)
{
$msg = 'Invalid username or password';
}?>
<html><body><?php echo $msg; ?><form>...</form></body></html>
這樣的代碼很常見。使用變量來顯示狀態信息有一定的局限性。因為它們無法通過重定向發送(除非你將它們作為GET變量傳播給下一個腳本,但這非常愚蠢)。而且在大型腳本中可能會有多個消息等。
最好的辦法是使用會話來傳播(即使是在同一頁面上)。想要這樣做的話在每個頁面上必須得有一個session_start。
function set_flash($msg){
$_SESSION['message'] = $msg;
}
function get_flash(){
$msg = $_SESSION['message'];
unset($_SESSION['message']);
return $msg;
}
在你的腳本中:
<?phpif($wrong_username || $wrong_password)
{
set_flash('Invalid username or password');
}
?><html><body>Status is : <?php echo get_flash(); ?><form>...</form></body></html>
5.讓函數變得靈活
function add_to_cart($item_id , $qty){
$_SESSION['cart'][$item_id] = $qty;
}
add_to_cart( 'IPHONE3' , 2 );
當添加單一條目時,使用上面的函數。那么當添加多個條目時,就得創建另一個函數嗎?NO。只要讓函數變得靈活起來使之能夠接受不同的參數即可。請看:
function add_to_cart($item_id , $qty){
if(!is_array($item_id))
{
$_SESSION['cart'][$item_id] = $qty;
}
else
{
foreach($item_id as $i_id => $qty)
{
$_SESSION['cart'][$i_id] = $qty;
}
}
}
add_to_cart( 'IPHONE3' , 2 );
add_to_cart( array('IPHONE3' => 2 , 'IPAD' => 5) );
好了,現在同樣的函數就可以接受不同類型的輸出了。以上代碼可以應用到很多地方讓你的代碼更加靈活。
6.省略結束的php標簽,如果它是腳本中的最后一行
我不知道為什么很多博客文章在談論php小技巧時要省略這個技巧。
<?php
echo "Hello";//Now dont close this tag
這可以幫助你省略大量問題。舉一個例子:
類文件super_class.php
<?php
class super_class{
function super_function()
{
//super code
}
}
?>//super extra character after the closing tag
現在看index.php
require_once('super_class.php');//echo an image or pdf , or set the cookies or session data
你會得到發送錯誤的Header。為什么呢?因為“超級多余字符”,所有標題都去處理這個去了。于是你得開始調試。你可能需要浪費很多時間來尋找超級額外的空間。
因此要養成省略結束標簽的習慣:
<?php
class super_class{
function super_function()
{
//super code
}
}//No closing tag
這樣更好。
7.在一個地方收集所有輸出,然后一次性輸出給瀏覽器
這就是所謂的輸出緩沖。比方說,你從不同的函數得到像這樣的內容:
function print_header(){
echo "<div id='header'>Site Log and Login links</div>";
}
function print_footer(){
echo "<div id='footer'>Site was made by me</div>";
}
print_header();
for($i = 0 ; $i < 100; $i++)
{
echo "I is : $i <br />';
}
print_footer();
}
print_footer();
其實你應該先在一個地方收集所有輸出。你可以要么將它存儲于函數中的變量內部,要么使用ob_start和ob_end_clean。所以,現在應該看起來像這樣
function print_header(){
$o = "<div id='header'>Site Log and Login links</div>";
return $o;
}
function print_footer(){
$o = "<div id='footer'>Site was made by me</div>";
return $o;
}
echo print_header();
for($i = 0 ; $i < 100; $i++)
{
echo "I is : $i <br />';
}
echo print_footer();
}
echo print_footer();
那么,為什么你應該做輸出緩沖呢:
你可以在將輸出發送給瀏覽器之前更改它,如果你需要的話。例如做一些str_replaces,或者preg_replaces,又或者是在末尾添加一些額外的html,例如profiler/debugger輸出。發送輸出給瀏覽器,并在同一時間做php處理并不是好主意。你見過這樣的網站,它有一個Fatal error在側邊欄或在屏幕中間的方框中嗎?你知道為什么會出現這種情況嗎?因為處理過程和輸出被混合在了一起。
8.當輸出非HTML內容時,通過header發送正確的mime類型
請看一些XML。
$xml = '<?xml version="1.0" encoding="utf-8" standalone="yes"?>';
$xml = "<response>
<code>0</code></response>";
//Send xml data
echo $xml;
工作正常。但它需要一些改進。
$xml = '<?xml version="1.0" encoding="utf-8" standalone="yes"?>';
$xml = "<response>
<code>0</code></response>";
//Send xml data
header("content-type: text/xml");
echo $xml;
請注意header行。這行代碼告訴瀏覽器這個內容是XML內容。因此,瀏覽器能夠正確地處理它。許多JavaScript庫也都依賴于header信息。
JavaScript,css,jpg圖片,png圖像也是一樣:
JavaScript
header("content-type: application/x-javascript");
echo "var a = 10";
CSSheader("content-type: text/css");
echo "#div id { background:#000; }"
9.為MySQL連接設置正確的字符編碼
曾碰到過unicode/utf-8字符被正確地存儲在mysql表的問題,phpmyadmin也顯示它們是正確的,但是當你使用的時候,你的網頁上卻并不能正確地顯示。里面的奧妙在于MySQL連接校對。
$host = 'localhost';$username = 'root';
$password = 'super_secret';
//Attempt to connect to database
$c = mysqli_connect($host , $username, $password);
//Check connection validity
if (!$c)
{
die ("Could not connect to the database host: <br />". mysqli_connect_error());
}
//Set the character set of the connection
if(!mysqli_set_charset ( $c , 'UTF8' ))
{
die('mysqli_set_charset() failed');
}
一旦你連接到數據庫,不妨設置連接字符集。當你在你的應用程序中使用多種語言時,這絕對有必要。
否則會發生什么呢?你會在非英文文本中看到很多的方框和????????。
10.使用帶有正確字符集選項的htmlentities
PHP 5.4之前,使用的默認字符編碼是ISO-8859-1,這不能顯示例如à a 這樣的字符。
$value = htmlentities($this->value , ENT_QUOTES , 'UTF-8');
從PHP 5.4起,默認編碼成了UTF-8,這解決了大部分的問題,但你最好還是知道這件事,如果你的應用程序使用多種語言的話。
11.不要在你的應用程序中gzip輸出,讓apache來做
考慮使用ob_gzhandler?不,別這樣做。它沒有任何意義。PHP應該是來寫應用程序的。不要擔心PHP中有關如何優化在服務器和瀏覽器之間傳輸的數據。
使用apache mod_gzip/mod_deflate通過.htaccess文件壓縮內容。
12.從php echo javascript代碼時使用json_encode
有些時候一些JavaScript代碼是從php動態生成的。
$images = array( 'myself.png' , 'friends.png' , 'colleagues.png');
$js_code = '';
foreach($images as $image)
{
$js_code .= "'$image' ,";
}
$js_code = 'var images = [' . $js_code . ']; ';
echo $js_code;
//Output is var images = ['myself.png' ,'friends.png' ,'colleagues.png' ,];
放聰明點。使用json_encode:
$images = array( 'myself.png' , 'friends.png' , 'colleagues.png');
$js_code = 'var images = ' . json_encode($images);echo $js_code;
//Output is : var images = ["myself.png","friends.png","colleagues.png"]
這不是很整潔?
13.在寫入任何文件之前檢查目錄是否可寫
在寫入或保存任何文件之前,請務必要檢查該目錄是否是可寫的,如果不可寫的話,會閃爍錯誤消息。這將節省你大量的“調試”時間。當你工作于Linux時,權限是必須要處理的,并且會有很多很多的權限問題時,當目錄不可寫,文件無法讀取等的時候。
請確保你的應用程序盡可能智能化,并在最短的時間內報告最重要的信息。
$contents = "All the content";
$file_path = "/var/www/project/content.txt";
file_put_contents($file_path , $contents);
這完全正確。但有一些間接的問題。file_put_contents可能會因為一些原因而失敗:
父目錄不存在
目錄存在,但不可寫
鎖定文件用于寫入?
因此,在寫入文件之前最好能夠一切都弄明確。
$contents = "All the content";
$dir = '/var/www/project';
$file_path = $dir . "/content.txt";if(is_writable($dir))
{
file_put_contents($file_path , $contents);
}else{
die("Directory $dir is not writable, or does not exist. Please check");
}
通過這樣做,你就能得到哪里文件寫入失敗以及為什么失敗的準確信息。
14.改變應用程序創建的文件的權限
當在Linux環境下工作時,權限處理會浪費你很多時間。因此,只要你的php應用程序創建了一些文件,那就應該修改它們的權限以確保它們在外面“平易近人”。否則,例如,文件是由“php”用戶創建的,而你作為一個不同的用戶,系統就不會讓你訪問或打開文件,然后你必須努力獲得root權限,更改文件權限等等。
// Read and write for owner, read for everybody elsechmod("/somedir/somefile", 0644);// Everything for owner, read and execute for otherschmod("/somedir/somefile", 0755);
15.不要檢查提交按鈕值來檢查表單提交
if($_POST['submit'] == 'Save')
{
//Save the things
}
以上代碼在大多數時候是正確的,除了應用程序使用多語言的情況。然后“Save”可以是很多不同的東西。那么你該如何再做比較?所以不能依靠提交按鈕的值。相反,使用這個:
if( $_SERVER['REQUEST_METHOD'] == 'POST' and isset($_POST['submit']) )
{
//Save the things
}
現在你就可以擺脫提交按鈕的值了。
16.在函數中總是有相同值的地方使用靜態變量
//Delay for some time
function delay(){
$sync_delay = get_option('sync_delay');
echo "<br />Delaying for $sync_delay seconds...";
sleep($sync_delay);
echo "Done <br />";
}
相反,使用靜態變量:
//Delay for some time
function delay(){
static $sync_delay = null;
if($sync_delay == null)
{
$sync_delay = get_option('sync_delay');
}
echo "<br />Delaying for $sync_delay seconds...";
sleep($sync_delay);
echo "Done <br />";
}
17.不要直接使用$ _SESSION變量
一些簡單的例子是:
$_SESSION['username'] = $username;$username = $_SESSION['username'];
但是這有一個問題。如果你正在相同域中運行多個應用程序,會話變量會發生沖突。2個不同的應用程序在會話變量中可能會設置相同的鍵名。舉個例子,一個相同域的前端門戶和后臺管理應用程序。
因此,用包裝函數使用應用程序特定鍵:
define('APP_ID' , 'abc_corp_ecommerce');//Function to get a session variable
function session_get($key){
$k = APP_ID . '.' . $key;
if(isset($_SESSION[$k]))
{
return $_SESSION[$k];
}
return false;
}//Function set the session variable
function session_set($key , $value){
$k = APP_ID . '.' . $key;
$_SESSION[$k] = $value;
return true;
}
18.封裝實用輔助函數到一個類中
所以,你必須在一個文件中有很多實用函數:
function utility_a(){ //This function does a utility thing like string processing}
function utility_b(){ //This function does nother utility thing like database processing}
function utility_c(){ //This function is ...}
自由地在應用程序中使用函數。那么你或許想要將它們包裝成一個類作為靜態函數:
class Utility{ public static function utility_a()
{
} public static function utility_b()
{
} public static function utility_c()
{
}
}//and call them as $a = Utility::utility_a();$b = Utility::utility_b();
這里你可以得到的一個明顯好處是,如果php有相似名稱的內置函數,那么名稱不會發生沖突。
從另一個角度看,你可以在相同的應用程序中保持多個版本的相同類,而不會發生任何沖突。因為它被封裝了,就是這樣。
19.一些傻瓜式技巧
使用echo代替print
使用str_replace代替preg_replace,除非你確定需要它
不要使用short tags
對于簡單的字符串使用單引號代替雙引號
在header重定向之后要記得做一個exit
千萬不要把函數調用放到for循環控制行中。
isset比strlen快
正確和一致地格式化你的代碼
不要丟失循環或if-else塊的括號。
不要寫這樣的代碼:
if($a == true) $a_count++;
這絕對是一種浪費。
這樣寫
if($a == true)
{
$a_count++;
}
不要通過吃掉語法縮短你的代碼。而是要讓你的邏輯更簡短。
使用具有代碼高亮功能的文本編輯器。代碼高亮有助于減少錯誤。
20. 使用array_map快速處理數組
比方說,你要trim一個數組的所有元素。新手會這樣做:
foreach($arr as $c => $v)
{
$arr[$c] = trim($v);
}
但它可以使用array_map變得更整潔:
$arr = array_map('trim' , $arr);
這適用于trim數組$arr的所有元素。另一個類似的函數是array_walk。
21.使用php過濾器驗證數據
你是不是使用正則表達式來驗證如電子郵件,IP地址等值?是的,每個人都是這樣做的。現在,讓我們試試一個不同的東西,那就是過濾器。
php過濾器擴展程序將提供簡單的方法來有效驗證或校驗值。
22.強制類型檢查
$amount = intval( $_GET['amount'] );$rate = (int) $_GET['rate'];
這是一種好習慣。
23.使用set_error_handler()將Php錯誤寫入到文件
set_error_handler()可以用來設置自定義的錯誤處理程序。在文件中編寫一些重要的錯誤用于日志是個好主意。
24.小心處理大型數組
大型的數組或字符串,如果一個變量保存了一些規模非常大的東西,那么要小心處理。常見錯誤是創建副本,然后耗盡內存,并得到內存溢出的致命錯誤:
$db_records_in_array_format; //This is a big array holding 1000 rows from a table each having 20 columns , every row is atleast 100 bytes , so total 1000 * 20 * 100 = 2MB$cc = $db_records_in_array_format; //2MB moresome_function($cc); //Another 2MB ?
當導入csv文件或導出表到csv文件時,上面這樣的代碼很常見。
像上面這樣做可能經常會由于內存限制而讓腳本崩潰。對于小規模的變量它不會出現問題,但當處理大型數組時一定要對此加以避免。
考慮通過引用傳遞它們,或者將它們存儲在一個類變量中:
$a = get_large_array();
pass_to_function(&$a);
這樣一來,相同的變量(并非其副本)將用于該函數。
class A{
function first()
{
$this->a = get_large_array();
$this->pass_to_function();
}
function pass_to_function()
{
//process $this->a
}
}
盡快復原它們,這樣內存就能被釋放,并且腳本的其余部分就能放松。
下面是關于如何通過引用來賦值從而節省內存的一個簡單示例。
<?phpini_set('display_errors' , true);
error_reporting(E_ALL);
$a = array();
for($i = 0; $i < 100000 ; $i++)
{
$a[$i] = 'A'.$i;
}
echo 'Memory usage in MB : '. memory_get_usage() / 1000000 . '<br />';
$b = $a;
$b[0] = 'B';
echo 'Memory usage in MB after 1st copy : '. memory_get_usage() / 1000000 . '<br />';
$c = $a;
$c[0] = 'B';
echo 'Memory usage in MB after 2st copy : '. memory_get_usage() / 1000000 . '<br />';
$d =& $a;
$d[0] = 'B';
echo 'Memory usage in MB after 3st copy (reference) : '. memory_get_usage() / 1000000 . '<br />';
一個典型php 5.4機器上的輸出是:
Memory usage in MB : 18.08208Memory usage in MB after 1st copy : 27.930944Memory usage in MB after 2st copy : 37.779808Memory usage in MB after 3st copy (reference) : 37.779864
因此可以看出,內存被保存在第3份通過引用的副本中。否則,在所有普通副本中內存將被越來越多地使用。
25.在整個腳本中使用單一的數據庫連接
請確保你在整個腳本使用單一的數據庫連接。從一開始就打開連接,使用至結束,并在結束時關閉它。不要像這樣在函數內打開連接:
function add_to_cart(){
$db = new Database();
$db->query("INSERT INTO cart .....");
}
function empty_cart(){
$db = new Database();
$db->query("DELETE FROM cart .....");
}
有多個連接也不好,會因為每個連接都需要時間來創建和使用更多的內存,而導致執行減緩。
在特殊情況下。例如數據庫連接,可以使用單例模式。
26.避免直接寫SQL,要把SQL語句抽象出來
不厭其煩的寫了太多如下的語句:
$query = "INSERT INTO users(name , email , address , phone) VALUES('$name' , '$email' , '$address' , '$phone')";$db->query($query); //call to mysqli_query()
這不是個建壯的方案。它有些缺點:
每次都手動轉義值
驗證查詢是否正確
查詢的錯誤會花很長時間識別(除非每次都用if-else檢查)
很難維護復雜的查詢
因此使用函數封裝:
function insert_record($table_name , $data){
foreach($data as $key => $value)
{ //mysqli_real_escape_string
$data[$key] = $db->mres($value);
}
$fields = implode(',' , array_keys($data));
$values = "'" . implode("','" , array_values($data)) . "'";
//Final query
$query = "INSERT INTO {$table}($fields) VALUES($values)";
return $db->query($query);
}
$data = array('name' => $name , 'email' => $email , 'address' => $address , 'phone' => $phone);
insert_record('users' , $data);
看到了嗎?這樣會更易讀和擴展。record_data函數小心的處理了轉義。
最大的優點是數據被預處理為一個數組,任何語法錯誤都會被捕獲。
該函數應該定義在某個database類中,你可以像$db->insert_record這樣調用。
類似的也可以編寫update,select,delete方法。試試吧。
27.將數據庫生成的內容緩存到靜態文件中
如果所有的內容都是從數據庫獲取的,它們應該被緩存。一旦生成了,就將它們保存在臨時文件中。下次請求該頁面時,可直接從緩存中取,不用再查數據庫。
好處:
節約php處理頁面的時間, 執行更快
更少的數據庫查詢意味著更少的mysql連接開銷
28.在數據庫中保存session
基于文件的session策略會有很多限制。使用基于文件的session不能擴展到集群中,因為session保存在單個服務器中。但數據庫可被多個服務器訪問。這樣就可以解決問題。在數據庫中保存session數據,還有更多好處:
處理username重復登錄問題。同個username不能在兩個地方同時登錄。
能更準備的查詢在線用戶狀態。
29. 避免使用全局變量
使用defines/constants
使用函數獲取值
使用類并通過$this訪問
30.在head中使用base標簽
沒聽說過?請看下面:
<head><base href="http://www.domain.com/store/"></head><body><img src="happy.jpg" /></body></html>
base標簽非常有用。假設你的應用分成幾個子目錄,它們都要包括相同的導航菜單。
www.domain.com/store/home.php
www.domain.com/store/products/ipad.php
在首頁中,可以寫:
<a href="home.php">Home</a><a href="products/ipad.php">Ipad</a>
但在你的ipad.php不得不寫成:
<a href="../home.php">Home</a><a href="ipad.php">Ipad</a>
因為目錄不一樣。有這么多不同版本的導航菜單要維護,很糟糕啊。因此,請使用base標簽。
<head><base href="http://www.domain.com/store/"></head><body><a href="home.php">Home</a><a href="products/ipad.php">Ipad</a></body></html>
現在,這段代碼放在應用的各個目錄文件中行為都一致。
31.永遠不要將error_reporting設為0
關閉不相的錯誤報告. E_FATAL 錯誤是很重要的.
ini_set('display_errors', 1);
error_reporting(~E_WARNING & ~E_NOTICE & ~E_STRICT);
注意:
這個讓web服務器記錄錯誤信息的文件'/path/to/errors.txt'應該讓web服務器具有可寫的權限。
這個錯誤日志文件要獨立開來。否則所有的各種日志,包括apache web服務器的日志、其它的錯誤日志都將混雜在一起了。
而且,為當前應用程序設置的錯誤日志文件應該只記錄了當前應用程序的錯誤日志(有可能web服務器上還運行了其它的應用程序)。
應用的錯誤日志應該放在當前應用程序的某個目錄下,使得像/var/log這樣的系統目錄不再需要去尋找。
不要設置error_reporting為0。這會使得所有發生的一切都不會被記錄。
另外,set_error_handler應該被用于設置一個用戶自定義的錯誤處理方法。例如,這個特有的功能,可以記錄所有的錯誤到一個文件中。
在開發環境的php.ini 中設置'display_errors=On'
在開發環境的php.ini里,啟用display_errors的權限是很重要的 (且不要依賴于 ini_set 的設置)。這是因為任何編譯時發生的錯誤都不允許ini_set來運行,這會導致出現一個空白頁面而沒有任何錯誤信息被顯示出來。
同樣的,在php.ini中設置為了On,但在代碼中把其設置為了off,那么錯誤發生時同樣顯示不出來。
在產品環境的php.ini中設置'display_errors=Off'
不要依賴于代碼init_set('display_errors',0);因為如果編譯時代碼發生錯誤,那么這條語句就不會被執行,而錯誤信息就會馬上顯示給了客戶。
32.注意平臺體系結構
integer在32位和64位體系結構中長度是不同的。因此某些函數如strtotime的行為會不同。在64位的機器中,你會看到如下的輸出。
$ php -aInteractive shell
php > echo strtotime("0000-00-00 00:00:00");
-62170005200php > echo strtotime('1000-01-30');
-30607739600php > echo strtotime('2100-01-30');4104930600
但在32位機器中, 它們將是bool(false)。
33.不要過分依賴set_time_limit
如果你想限制最小時間,可以使用下面的腳本:
set_time_limit(30);
//Rest of the code
高枕無憂嗎?注意任何外部的執行,如系統調用,socket操作,數據庫操作等,就不在set_time_limits的控制之下。因此,就算數據庫花費了很多時間查詢,腳本也不會停止執行。視情況而定。
34.使用擴展庫
一個應用常常需要使用到很多用基本php代碼實現的功能。像生成pdf文件,處理圖片,發送郵件,生成圖形和文檔等等。有大量的外部庫能夠快速輕易地實現這些功能。下面是一些很流行的庫:
mPDF - 生成pdf文檔,能漂亮地把html轉換為pdf格式。
PHPExcel - 讀寫Excel文件
PhpMailer - 容易地實現發送帶附件的html郵件
pChart - 用php生成圖片
35.使用MVC框架
是時候使用像codeigniter這樣的MVC框架了。MVC框架并不強迫你寫面向對象的代碼。它們僅將php代碼與html分離。
明確區分php和html代碼。在團隊協作中有好處,設計師和程序員可以同時工作。
面向對象設計的函數能讓你更容易維護
內建函數完成了很多工作。你不需要重復編寫
開發大的應用是必須的
很多建議,技巧和hack已被框架實現了
36.時常看看phpbench
phpbench提供了些php基本操作的基準測試結果,它展示了一些徽小的語法變化是怎樣導致巨大差異的。
37.閱讀php官方文檔
PHP官方網站有每一個功能、類和及其方法的文檔。所有的這些文檔下面都有這大量的用戶反饋,并包含了來自于社區的大量有價值的信息。其中有用戶的反饋,專家建議和代碼片段等。因此,去看一下吧。
38.到IRC頻道去問問題
IRC頻道:#php,是一個最好的在線問php相關問題的地方。雖然有了大量的博客和論壇的文章,且每天還在不斷地增加,但當一個特殊的問題產生時,在博客和論壇上面未必就能找的。那么,這時IRC的php頻道正是那個你要問問題的地方。而且,還都是免費的!
39.閱讀開源項目
閱讀其它開源項目代碼一直是一個提升自己能力的好辦法。學習里面的技術、代碼風格、注釋風格、代碼組織和文件命名等。我閱讀的第一個開源項目是codeigniter框架。這個框架很容易使用,也很容易讀懂。下面是我推薦的其它的一些開源項目
Codeigniter
WordPress
Joomla CMS
- PHP學習
- PHP應用
- PHP函數總結整理
- 39個對初學者非常有用的PHP技巧
- 深入淺出之Smarty模板引擎工作機制
- 數組操作
- file操作的常用方法
- PHP字符串輸出之Heredoc說明
- require(_once)和include(_once)的理解
- PHP提高效率的幾點
- php無限遍歷目錄
- 53個要點提高PHP編程效率
- THINKPHP
- THINKPHP 常見的問題
- 微信
- 微信公眾號接口
- 微信小程序開發資料收集
- 微信小程序開發:MINA
- 通過微信小程序看前端
- 微信小程序開發初體驗
- 微信小程序 Demo(豆瓣電影)
- API應用
- 支付寶
- 二維碼轉換
- 前端開發
- HTML5
- CSS
- 七種css方式讓一個容器水平垂直居中
- JavaScript
- JavaScript奇技淫巧44招
- JavaScript筆記
- 后端開發
- Node
- SQL數據庫
- 服務維護
- git使用
- Git入門私房菜
- MAC終端維護
- VIM命令大全
- 開發規范
- 智能手機屏幕的秘密
- 超實用六步透視網易設計規范(附完整PDF下載)
- UI設計常用字體規范
- APP界面切圖命名和文件整理規范
- 網頁UI視覺設計規范
- ios視覺設計規范說明
- 開發APP時需要注意的原則
- 移動端APP設計初步入門
- Axure
- 基礎操作
- 基礎1-10
- 基礎11-20
- 基礎21-30
- 基礎31-40
- 基礎41-50
- 基礎51-60
- Sketch
- 軟件使用
- sublime3_用戶設置
- sublime下如何修改自動補全 后lang=zh-cn?
- 運營理念
- 新人指導心得體會
- 從一次活動設計,聊聊交互設計師的3個階段
- 詳情頁優化那些不得不說的細節
- 店鋪裝修淺析
- 淘寶店鋪裝修之寶貝詳情頁的布局教程
- 寶貝詳情頁 客戶需求調研及總結
- 寶貝描述樣板房