php代碼優化

1 代碼優化

 

1 儘可能靜態化

若是一個方法能被靜態,那就聲明它爲靜態的,速度可提升1/4,甚至我測試的時候,這個提升了近三倍。php

固然了,這個測試方法須要在十萬級以上次執行,效果才明顯。css

其實靜態方法和非靜態方法的效率主要區別在內存:靜態方法在程序開始時生成內存,實例方法(非靜態方法)在程序運行中生成內存,因此靜態方法能夠直接調用,實例方法要先成生實例再調用,靜態速度很快,可是多了會佔內存。html

任何語言都是對內存和磁盤的操做,至因而否面向對象,只是軟件層的問題,底層都是同樣的,只是實現方法不一樣。靜態內存是連續的,由於是在程序開始時就生成了,而實例方法申請的是離散的空間,因此固然沒有靜態方法快。python

靜態方法始終調用同一塊內存,其缺點就是不能自動進行銷燬,而實例化能夠銷燬。mysql

2 echo效率高於print

由於echo沒有返回值,print返回一個整型。測試:nginx

echo
0.000929 - 0.001255 s (平均 0.001092 seconds) print 0.000980 - 0.001396 seconds (平均 0.001188 seconds)

相差8%左右,整體上echo是比較快的。web

注意:echo輸出大字符串的時候,若是沒有調整就會嚴重影響性能。打開Apache的mod_deflate進行壓縮,或者打開ob_start將內容放進緩衝區能夠改善性能問題。正則表達式

3 循環最大次數

在循環以前設置循環的最大次數,而非在在循環中。算法

4 及時銷燬變量

數組和對象在 PHP 中特別佔內存的,這個因爲 PHP 的底層的zend引擎引發的。通常來講,PHP數組的內存利用率只有 1/10, 也就是說,一個在C語言裏面100M 內存的數組,在PHP裏面就要1G。sql

特別是在PHP做爲後臺服務器的系統中,常常會出現內存耗費太大的問題。

5 避免使用像__get、__set、__autoload等魔術方法

(僅供參考,有待商榷)

對於__開頭的函數就命名爲魔術函數,此類函數都在特定的條件下觸發的。總得來講,有下面幾個魔術函數__construct()__destruct()__get()__set()__unset()__call()__callStatic()__sleep()__wakeup()__toString()__set_state()__clone()__autoload()

其實,若是__autoload() 不能高效的將類名與實際的磁盤文件(注意,這裏指實際的磁盤文件,而不只僅是文件名)對應起來,系統將不得不作大量的文件是否存在判斷(須要在每一個include path中包含的路徑中去尋找),而判斷文件是否存在須要作磁盤I/O操做,衆所周知磁盤I/O操做的效率很低,所以這纔是使得autoload機制效率下降的緣由。

所以,咱們在系統設計時,須要定義一套清晰的將類名與實際磁盤文件映射的機制。這個規則越簡單越明確,autoload機制的效率就越高。

結論:autoload機制並非自然的效率低下,只有濫用autoload,設計很差的自動裝載函數纔會致使其效率的下降.

因此說盡可能避免使用__autoload魔術方法,有待商榷。

6 requiere_once() 和 include_once() 比較耗資源

這是由於requiere_once()include_once()須要判斷該文件是否被引用過,因此能不用盡可能不用。經常使用require/include方法避免。鳥哥在其博客中就屢次聲明儘可能不要用require_onceinclude_once

7 在include和require中使用絕對路徑

若是包含相對路徑,PHP會在include_path裏面遍歷查找文件。
用絕對路徑就會避免此類問題,所以解析操做系統路徑所需的時間會更少。

8 使用$_SERVER['REQUSET_TIME']

若是你須要獲得腳本執行的時間,$_SERVER['REQUSET_TIME']優於time()

能夠想象,一個是現成就能夠直接用,一個還須要函數得出的結果。

9 用內置函數替代正則表達式

能用PHP內部字符串操做函數的狀況下,儘可能用他們,不要用正則表達式, 由於其效率高於正則。

沒得說,正則最耗性能。

有沒有你漏掉的好用的函數?例如:strpbrk()strncasecmp()strpos()strrpos()stripos()strripos()

strtr() 函數用於轉換指定字符,若是須要轉換的全是單個字符的時候,用字符串而不是數組:

<?php $addr = strtr($addr, "abcd", "efgh"); // good $addr = strtr($addr, array('a' => 'e', )); // bad

效率提高:10 倍。

10 用strtr做字符替換

str_replace字符替換比正則替換preg_replace快,但strtrstr_replace又快1/4。

另外,不要作無謂的替換,即便沒有替換,str_replace也會爲其參數分配內存。很慢!

解決辦法:用 strpos 先查找(很是快),看是否須要替換,若是須要,再替換。

效率:若是須要替換,效率幾乎相等,差異在 0.1% 左右。若是不須要替換:用 strpos 快 200%

11 用字符串而不是數組做爲參數

若是一個函數既能接受數組,又能接受簡單字符作爲參數,那麼儘可能用字符做爲參數。例如字符替換函數,參數列表並非太長,就能夠考慮額外寫一段替換代碼,使得每次傳遞參數都是一個字符,而不是接受數組作爲查找和替換參數。大事化小,1+1>2

12 最好不用@

@掩蓋錯誤會下降腳本運行速度,而且在後臺有不少額外操做。用@比起不用,效率差距 3 倍。特別不要在循環中使用@,在 5 次循環的測試中,即便是先用error_reporting(0)關掉錯誤,在循環完成後再打開,都比用@快。

13 數組元素加引號

$row['id']$row[id]速度快7倍,建議養成數組鍵名加引號的習慣。

14 別在循環裏用函數

例如:

for($x=0; $x < count($array); $x++) { }

這種寫法在每次循環的時候都會調用 count() 函數,效率大大下降,建議這樣:

$len = count($array); for($x=0; $x < $len; $x++) { }

讓函數在循環外面一次得到循環次數。

16 方法裏創建局部變量

在類的方法裏創建局部變量速度最快,幾乎和在方法裏調用局部變量同樣快。

17 局部變量比全局變量快2倍

因爲局部變量是存在棧中的,當一個函數佔用的棧空間不是很大的時候,這部份內存頗有可能所有命中cache,這時候CPU訪問的效率是很高的。
相反,若是一個函數裏既使用了全局變量又使用了局部變量,那麼當這兩段地址相差較大時,cpu cache須要來回切換,那麼效率會降低。

18 局部變量而不是對象屬性

創建一個對象屬性(類裏面的變量,例如:$this->prop++)比局部變量要慢3倍。

19 提早聲明局部變量

創建一個未聲明的局部變量要比一個已經定義過的局部變量慢9-10倍。

20 謹慎聲明全局變量

聲明一個未被任何一個函數使用過的全局變量也會使性能下降(和聲明相同數量的局部變量同樣)。PHP可能去檢查這個全局變量是否存在。

21 類的性能和其方法數量沒有關係

新添加10個或多個方法到測試的類後,性能沒什麼差別。

22 在子類裏方法的性能優於在基類中

23 函數快於類方法

調用只有一個參數、而且函數體爲空的函數,花費的時間等於7-8次$localvar++運算,而同一功能的類方法大約爲15次$localvar++運算。

24 用單引號代替雙引號會快一些

由於PHP會在雙引號包圍的字符串中搜尋變量,單引號則不會。

PHP 引擎容許使用單引號和雙引號來封裝字符串變量,可是它們的速度是有很大的差異的!使用雙引號的字符串會告訴 PHP 引擎,首先去讀取字符串內容,查找其中的變量,並改成變量對應的值。通常來講字符串是沒有變量的,因此使用雙引號會致使性能不佳。最好是使用字符串鏈接而不是雙引號字符串。

$output = "This is a plain string";  // 很差的實踐 $output = 'This is a plain string'; // 好的實踐 $type = "mixed"; // 很差的實踐 $output = "This is a $type string"; $type = 'mixed'; // 好的實踐 $output = 'This is a ' . $type .' string';

25 echo字符串用逗號代替點鏈接符更快些

echo能夠把逗號隔開的多個字符串看成「函數」參數傳入,因此速度會更快。(說明:echo是一種語言結構,不是真正的函數,故把函數加上了雙引號)。例如:

echo $str1, $str2;       // 速度快 echo $str1 . $str2; // 速度稍慢

26 儘可能靜態化

Apache/Nginx解析一個PHP腳本的時間,要比解析一個靜態HTML頁面慢2至10倍,因此儘可能使頁面靜態化,或使用靜態HTML頁面。

28 使用緩存

Memchached或者Redis均可以。

高性能的分佈式內存對象緩存系統,提升動態網絡應用程序性能,減輕數據庫的負擔。

也對運算碼 (OP code)的緩存頗有用,使得腳本沒必要爲每一個請求作從新編譯。

29 使用整型保存IP

使用ip2long()long2ip()函數把IP地址轉成整型後,再存放進數據庫,而保存非字符型。

這幾乎能下降1/4的存儲空間。同時能夠很容易對地址進行排序和快速查找;

30 檢查email有效性

使用checkdnsrr()經過域名存在性來確認email地址的有效性,這個內置函數能保證每個的域名對應一個IP地址。

31 使用MySQLi或PDO

mysql_*函數已經不被建議使用,建議使用加強型的mysqli_*系列函數或者直接使用PDO。

32 試着喜歡使用三元運算符(?:)

33 是否須要組件

在你想在完全重作你的項目前,看看是否有現成的組件(在Packagist上)可用,經過composer安裝。組件是別人已經造好的輪子,是個巨大的資源庫,不少php開發者都知道。

35 屏蔽敏感信息

使用error_reporting()函數來預防潛在的敏感信息顯示給用戶。

理想的錯誤報告應該被徹底禁用在php.ini文件裏。但是若是你在用一個共享的虛擬主機,php.ini你不能修改,那麼你最好添加error_reporting()函數,放在每一個腳本文件的第一行(或用require_once()來加載)這能有效的保護敏感的SQL查詢和路徑在出錯時不被顯示;

36 壓縮大的字符串

使用gzcompress()gzuncompress()對容量大的字符串進行壓縮/解壓,再存進/取出數據庫。這種內置的函數使用gzip算法,能壓縮字符串90%。

37 引用傳遞參數

經過參數地址引用使函數有多個返回值,在參數變量前加個「&」表示按地址傳遞,而非按值傳遞。

38 徹底理解魔術引用和SQL注入的危險。

Fully understand 「magic quotes」 and the dangers of SQL injection. I’m hoping that most developers reading this are already familiar with SQL injection. However, I list it here because it’s absolutely critical to understand. If you’ve never heard the term before, spend the entire rest of the day googling and reading.

39 某些地方使用isset代替strlen

當操做字符串並須要檢驗其長度是否知足某種要求時,你想固然地會使用strlen()函數。此函數執行起來至關快,由於它不作任何計算,只返回在zval結構(C的內置數據結構,用於存儲PHP變量)中存儲的已知字符串長度。可是,因爲strlen()是函數,多多少少會有些慢,由於函數調用會通過諸多步驟,如字母小寫化(譯註:指函數名小寫化,PHP不區分函數名大小寫)、哈希查找,會跟隨被調用的函數一塊兒執行。在某些狀況下,你可使用isset()技巧加速執行你的代碼。

例如:

if (strlen($foo) < 5) { echo "Foo is too short"; } // 使用isset() if (!isset($foo{5})) { echo "Foo is too short"; }

40 使用++$i遞增

當執行變量$i的遞增或遞減時,$i++會比++$i慢一些。這種差別是PHP特有的,並不適用於其餘語言,因此請不要修改你的C或Java代碼並期望它們能當即變快,沒用的。++$i更快是由於它只須要3條指令(opcodes),$i++則須要4條指令。後置遞增實際上會產生一個臨時變量,這個臨時變量隨後被遞增。而前置遞增直接在原值上遞增。這是最優化處理的一種,正如Zend的PHP優化器所做的那樣。牢記這個優化處理不失爲一個好主意,由於並非全部的指令優化器都會作一樣的優化處理,而且存在大量沒有裝配指令優化器的互聯網服務提供商(ISPs)和服務器。

40 不要隨便複製變量

有時候爲了使 PHP 代碼更加整潔,一些 PHP 新手(包括我)會把預約義好的變量複製到一個名字更簡短的變量中,其實這樣作的結果是增長了一倍的內存消耗,只會使程序更加慢。試想一下,在下面的例子中,若是用戶惡意插入 512KB 字節的文字到文本輸入框中,這樣就會致使 1MB 的內存被消耗!

// 很差的實踐
$description = $_POST['description']; echo $description; // 好的實踐 echo $_POST['description'];

41 使用選擇分支語句

switchcase好於使用多個ifelse if語句,而且代碼更加容易閱讀和維護。

42 用file_get_contents替代file、fopen、feof、fgets

在能夠用file_get_contents()替代file()fopen()feof()fgets()等系列方法的狀況下,儘可能用file_get_contents(),由於他的效率高得多!可是要注意,file_get_contents()在打開一個URL文件時候的PHP版本問題。

43 儘可能的少進行文件操做,雖然PHP的文件操做效率也不低的

44 優化Select SQL語句

在可能的狀況下儘可能少的進行insertupdate操做(在update上,我被惡批過)。

45 儘量的使用PHP內部函數

46 循環內部不要聲明變量,尤爲是大變量:對象

這好像不僅是PHP裏面要注意的問題吧?

47 多維數組儘可能不要循環嵌套賦值

48 循環用foreach效率更高

儘可能用foreach代替whilefor循環

50 對global變量,應該用完就unset()掉

51 並非事必面向對象(OOP)

面向對象每每開銷很大,每一個方法和對象調用都會消耗不少內存。

52 方法不要細分得過多

仔細想一想你真正打算重用的是哪些代碼?

53 耗時函數考慮用C擴展的方式實現

若是在代碼中存在大量耗時的函數,你能夠考慮用C擴展的方式實現它們。

54 mod_deflate壓縮輸出

打開apache的mod_deflate模塊,能夠提升網頁的瀏覽速度。(提到過echo 大變量的問題)

5五、數據庫鏈接當使用完畢時應關掉,不要用長鏈接

5六、split比exploade快

split()
0.001813 - 0.002271 seconds (avg 0.002042 seconds) explode() 0.001678 - 0.003626 seconds (avg 0.002652 seconds)

以上都是關於,下面是從總體結構方面優化PHP性能:

2 總體結構優化PHP性能

 

1 將PHP升級到最新版

提升性能的最簡單的方式是不斷升級、更新PHP版本。

2 使用分析器

網站運行緩慢的緣由頗多,Web應用程序極其複雜,讓人撲朔迷離。而一種可能性在於PHP代碼自己。這個分析器能夠幫助你快速找出形成瓶頸的代碼,提升網站運行的整體性能。

Xdebug PHP extension提供了強大的功能,能夠用來調試,也能夠用來分析代碼。方便開發人員直接跟蹤腳本的執行,實時查看綜合數據。還能夠將這個數據導入到可視化的工具 KCachegrind中。

3 檢錯報告

PHP支持強大的檢錯功能,方便你實時檢查錯誤,從比較重要的錯誤到相對小的運行提示。總共支持13種獨立的報告級別,你能夠根據這些級別靈活匹配,生成用戶自定義的檢測報告。

4 利用PHP的擴展

一直以來,你們都在抱怨PHP內容太過繁雜,最近幾年來開發人員做出了相應的努力,移除了項目中的一些冗餘特徵。即使如此,可用庫以及其它擴展的數量仍是很可觀。甚至一些開發人員開始考慮實施本身的擴展方案。

5 PHP緩存,使用PHP加速器:APC

通常狀況下,PHP腳本被PHP引擎編譯後執行,會被轉換成機器語言,也稱爲操做碼。若是PHP腳本通過反覆編譯而獲得相同的結果,那爲何不徹底跳過編譯過程呢?

經過PHP加速器,你徹底能夠實現這一點,它緩存了PHP腳本編譯後的機器碼,容許代碼根據要求當即執行,而不通過繁瑣的編譯過程。

對PHP開發人員而言,目前提供了兩種可用的緩存方案,一種是APC(Alternative PHP Cache,可選PHP緩存),它是一個能夠經過PEAR安裝的開源加速器。另外一種流行的方案是Zend Server,它不只提供了操做碼緩存技術,也提供了相應頁面的緩存工具。

6 內存緩存

PHP一般在檢索和數據分析方面扮演着重要角色,這些操做可能會致使性能下降。實際上有些操做是徹底沒有必要的,特別是從數據庫中反覆檢索一些經常使用的靜態數據。不妨考慮一下短時間使用Redis或Memcached extension來緩存數據。Memcached的擴展緩存與libMemcached庫協同工做,在RAM中緩存數據,也容許用戶定義緩存的期限,有助於確保用戶信息的實時更新。

7 內容壓縮

幾乎全部的瀏覽器都支持Gzip的壓縮方式,gzip能夠下降80%的輸出,付出的代價是大概增長了10%的cpu計算量。可是賺到的是不只佔用的帶寬減小了,並且你的頁面加載會變得很快,優化了你的PHP站點性能。你能夠在PHP.ini中開啓它:

zlib.output_compression = On zlib.output_compression_level = (level)

level多是1-9之間的數字,你能夠設置不一樣的數字使得他適合你的站點。
若是你使用apache,你也能夠激活mod_gzip模塊,他是高度可定製的。

8 服務器緩存

主要是基於web反向代理的靜態服務器nginx和squid,還有apache2的mod_proxy和mod_cache模塊

9 數據庫優化,緩存等

經過配置數據庫緩存,如開啓QueryCache緩存,當查詢接收到一個和以前一樣的查詢, 服務器將會從查詢緩存種檢索結果,而不是再次分析和執行上次的查詢以及數據存儲過程,鏈接池技術等。

 

 

原文地址:https://www.cnblogs.com/Steven-shi/p/5897766.html

相關文章
相關標籤/搜索