《PHP編程:深入分析PHP優(yōu)化及注意事項》要點:
本文介紹了PHP編程:深入分析PHP優(yōu)化及注意事項,希望對您有用。如果有疑問,可以聯(lián)系我們。
PHP教程1.盡量靜態(tài)化:
PHP教程如果一個方法能被靜態(tài),那就聲明它為靜態(tài)的,速度可提高1/4,甚至我測試的時候,這個提高了近三倍.
PHP教程當然了,這個測試方法需要在十萬級以上次執(zhí)行,效果才明顯.
PHP教程其實靜態(tài)方法和非靜態(tài)方法的效率主要區(qū)別在內(nèi)存:靜態(tài)方法在程序開始時生成內(nèi)存,實例方法在程序運行中生成內(nèi)存,所以靜態(tài)方法可以直接調(diào)用,實例方法要先成生實例,通過實例調(diào)用方法,靜態(tài)速度很快,但是多了會占內(nèi)存.
PHP教程任何語言都是對內(nèi)存和磁盤的操作,至于是否面向?qū)ο?只是軟件層的問題,底層都是一樣的,只是實現(xiàn)方法不同.靜態(tài)內(nèi)存是連續(xù)的,因為是在程序開始時就生成了,而實例申請的是離散的空間,所以當然沒有靜態(tài)方法快.
PHP教程靜態(tài)方法始終調(diào)用同一塊內(nèi)存,其缺點就是不能自動進行銷毀,而是實例化可以銷毀.
PHP教程2.echo的效率高于print,因為echo沒有返回值,print返回一個整型;
PHP教程測試:
PHP教程
Echo
0.000929 - 0.001255 s (平均 0.001092 seconds)
Print
0.000980 - 0.001396 seconds (平均 0.001188 seconds)
PHP教程相差8%左右,總體上echo是比較快的.
PHP教程注意,echo大字符串的時候,如果沒有做調(diào)整就嚴重影響性能.使用打開apached的mod_deflate進行壓縮或者打開ob_start先將內(nèi)容放進緩沖區(qū).
PHP教程3.在循環(huán)之前設(shè)置循環(huán)的最大次數(shù),而非在在循環(huán)中;
PHP教程傻子都明白的道理.
PHP教程4.銷毀變量去釋放內(nèi)存,特別是大的數(shù)組;
PHP教程數(shù)組和對象在php特別占內(nèi)存的,這個由于php的底層的zend引擎引起的,
PHP教程一般來說,PHP數(shù)組的內(nèi)存利用率只有 1/10,?也就是說,一個在C語言里面100M 內(nèi)存的數(shù)組,在PHP里面就要1G.
PHP教程特別是在PHP作為后臺服務(wù)器的系統(tǒng)中,經(jīng)常會出現(xiàn)內(nèi)存耗費太大的問題.
PHP教程5.避免使用像__get, __set, __autoload等魔術(shù)方法;
PHP教程對于__開頭的函數(shù)就命名為魔術(shù)函數(shù),此類函數(shù)都在特定的條件下初訪的.總得來說,有下面幾個魔術(shù)函數(shù)
PHP教程__construct(),__destruct(),__get(),__set(),__unset(),__call(),__callStatic(),__sleep(),__wakeup(),__toString(),__set_state(),__clone(),__autoload()
PHP教程其實,如果__autoload不能高效的將類名與實際的磁盤文件(注意,這里指實際的磁盤文件,而不僅僅是文件名)對應(yīng)起來,系統(tǒng)將不得不做大量的文件是 否存在(需要在每個include?path中包含的路徑中去尋找)的判斷,而判斷文件是否存在需要做磁盤I/O操作,眾所周知磁盤I/O操作的效率很低,因此這才是使得autoload機制效率降低的原因.
PHP教程因此,我們在系統(tǒng)設(shè)計時,需要定義一套清晰的將類名與實際磁盤文件映射的機制.這個規(guī)則越簡單越明確,autoload機制的效率就越高.
PHP教程結(jié)論:autoload機制并不是天然的效率低下,只有濫用autoload,設(shè)計不好的自動裝載函數(shù)才會導(dǎo)致其效率的降低.
PHP教程所以說盡量避免使用__autoload魔術(shù)方法,有待商榷.
PHP教程6.requiere_once()比較耗資源;
PHP教程這是因為requiere_once需要判斷該文件是否被引用過),所以能不用盡量不用.常用require/include方法避免.
PHP教程7.在includes和requires中使用絕對路徑.
PHP教程如果包含相對路徑,PHP會在include_path里面遍歷查找文件.
PHP教程用絕對路徑就會避免此類問題,因此解析操作系統(tǒng)路徑所需的時間會更少.
PHP教程8.如果你需要得到腳本執(zhí)行時的時間,$_SERVER['REQUSET_TIME']優(yōu)于time();可以想象.一個是現(xiàn)成就可以直接用,一個還需要函數(shù)得出的結(jié)果.
PHP教程9.能用PHP內(nèi)部字符串操作函數(shù)的情況下,盡量用他們,不要用正則表達式; 因為其效率高于正則;
沒得說,正則最耗性能.
PHP教程有沒有你漏掉的好用的函數(shù)?例如:strpbrk()strncasecmp()strpos()/strrpos()/stripos()/strripos()加速 strtr如果需要轉(zhuǎn)換的全是單個字符的時候,
用字符串而不是數(shù)組來做 strtr:
PHP教程
<?php
$addr = strtr($addr, "abcd", "efgh"); // good
$addr = strtr($addr, array('a' => 'e', )); // bad
?>
PHP教程效率提升:10 倍.
PHP教程10.str_replace字符替換比正則替換preg_replace快,但strtr比str_replace又快1/4;
PHP教程另外不要做無謂的替換即使沒有替換,str_replace 也會為其參數(shù)分配內(nèi)存.很慢!解決辦法:
用 strpos 先查找(非常快),看是否需要替換,如果需要,再替換效率:- 如果需要替換:效率幾乎相等,差別在 0.1% 左右.
如果不需要替換:用 strpos 快 200%.
PHP教程11.參數(shù)為字符串
PHP教程如果一個函數(shù)既能接受數(shù)組又能接受簡單字符做為參數(shù),例如字符替換函數(shù),并且參數(shù)列表不是太長,可以考慮額外寫一段替換代碼,使得每次傳遞參數(shù)都是一 個字符,而不是接受數(shù)組做為查找和替換參數(shù).大事化小,1+1>2;
PHP教程12.最好不用@,用@掩蓋錯誤會降低腳本運行速度;
PHP教程用@實際上后臺有很多操作.用@比起不用@,效率差距:3 倍.特別不要在循環(huán)中使用@,在 5 次循環(huán)的測試中,即使是先用 error_reporting(0) 關(guān)掉錯誤,在循環(huán)完成后再打開,都比用@快.
PHP教程13.$row['id']比$row[id]速度快7倍
PHP教程建議養(yǎng)成數(shù)組鍵加引號的習(xí)慣;
PHP教程14.在循環(huán)里別用函數(shù)
PHP教程例如For($x=0; $x < count($array); $x), count()函數(shù)在外面先計算;原因你懂的.
PHP教程16.在類的方法里建立局部變量速度最快,幾乎和在方法里調(diào)用局部變量一樣快;
PHP教程17.建立一個全局變量要比局部變量要慢2倍;
PHP教程由于局部變量是存在棧中的,當一個函數(shù)占用的棧空間不是很大的時候,這部分內(nèi)存很有可能全部命中cache,這時候CPU訪問的效率是很高的.
相反,如果一個函數(shù)里既使用了全局變量又使用了局部變量,那么當這兩段地址相差較大時,cpu cache需要來回切換,那么效率會下降.
(我理解啊)
PHP教程18.建立一個對象屬性(類里面的變量)例如($this->prop++)比局部變量要慢3倍;
PHP教程19.建立一個未聲明的局部變量要比一個已經(jīng)定義過的局部變量慢9-10倍
PHP教程20.聲明一個未被任何一個函數(shù)使用過的全局變量也會使性能降低(和聲明相同數(shù)量的局部變量一樣).
PHP可能去檢查這個全局變量是否存在;
PHP教程21.方法的性能和在一個類里面定義的方法的數(shù)目沒有關(guān)系
PHP教程因為我添加10個或多個方法到測試的類里面(這些方法在測試方法的前后)后性能沒什么差異;
PHP教程22.在子類里方法的性能優(yōu)于在基類中;
PHP教程23.只調(diào)用一個參數(shù)并且函數(shù)體為空的函數(shù)運行花費的時間等于7-8次$localvar++運算,而一個類似的方法(類里的函數(shù))運行等于大約15次$localvar++運算;
PHP教程24 用單引號代替雙引號來包含字符串,這樣做會更快一些.
因為PHP會在雙引號包圍的字符串中搜尋變量,單引號則不會.
PHP教程PHP 引擎允許使用單引號和雙引號來封裝字符串變量,但是這個是有很大的差別的!使用雙引號的字符串告訴 PHP 引擎首先去讀取字符串內(nèi)容,查找其中的變 量,并改為變量對應(yīng)的值.一般來說字符串是沒有變量的,所以使用雙引號會導(dǎo)致性能不佳.最好是使用字
PHP教程符串連接而不是雙引號字符串.
PHP教程
BAD:
$output = "This is a plain string";
GOOD:
$output = 'This is a plain string';
BAD:
$type = "mixed";
$output = "This is a $type string";
GOOD:
$type = 'mixed';
$output = 'This is a ' . $type .' string';
PHP教程25.當echo字符串時用逗號代替點連接符更快些.
PHP教程echo一種可以把多個字符串當作參數(shù)的“函數(shù)”(譯注:PHP手冊中說echo是語言結(jié)構(gòu),不是真正的函數(shù),故把函數(shù)加上了雙引號).
例如echo $str1,$str2.
PHP教程26.Apache解析一個PHP腳本的時間要比解析一個靜態(tài)HTML頁面慢2至10倍.
PHP教程盡量多用靜態(tài)HTML頁面,少用腳本.
PHP教程28.盡量使用緩存,建議用memcached.
PHP教程高性能的分布式內(nèi)存對象緩存系統(tǒng),提高動態(tài)網(wǎng)絡(luò)應(yīng)用程序性能,減輕數(shù)據(jù)庫的負擔;
也對運算碼 (OP code)的緩存很有用,使得腳本不必為每個請求做重新編譯.
PHP教程29.使用ip2long()和long2ip()函數(shù)把IP地址轉(zhuǎn)成整型存放進數(shù)據(jù)庫而非字符型.
PHP教程這幾乎能降低1/4的存儲空間.同時可以很容易對地址進行排序和快速查找;
PHP教程30.使用checkdnsrr()通過域名存在性來確認部分email地址的有效性
PHP教程這個內(nèi)置函數(shù)能保證每一個的域名對應(yīng)一個IP地址;
PHP教程31.使用mysql_*的改良函數(shù)mysqli_*;
PHP教程32.試著喜歡使用三元運算符(?:);
PHP教程33.是否需要PEAR
PHP教程在你想在徹底重做你的項目前,看看PEAR有沒有你需要的.PEAR是個巨大的資源庫,很多php開發(fā)者都知道;
PHP教程35.使用error_reporting(0)函數(shù)來預(yù)防潛在的敏感信息顯示給用戶.
PHP教程理想的錯誤報告應(yīng)該被完全禁用在php.ini文件里.可是如果你在用一個共享的虛擬主機,php.ini你不能修改,那么你最好添加error_reporting(0)函數(shù),放在每個腳本文件的第一行(或用
require_once()來加載)這能有效的保護敏感的SQL查詢和路徑在出錯時不被顯示;
PHP教程36.使用?gzcompress() 和gzuncompress()對容量大的字符串進行壓縮(解壓)在存進(取出)數(shù)據(jù)庫時.
PHP教程這種內(nèi)置的函數(shù)使用gzip算法能壓縮到90%;
PHP教程37.通過參數(shù)變量地址得引用來使一個函數(shù)有多個返回值.
PHP教程你可以在變量前加個“&”來表示按地址傳遞而非按值傳遞;
PHP教程38. 完全理解魔術(shù)引用和SQL注入的危險.
PHP教程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.
PHP教程39.某些地方使用isset代替strlen
PHP教程當操作字符串并需要檢驗其長度是否滿足某種要求時,你想當然地會使用strlen()函數(shù).此函數(shù)執(zhí)行起來相當快,因為它不做任何計算,只返回在zval 結(jié)構(gòu)(C的內(nèi)置數(shù)據(jù)結(jié)構(gòu),用于存儲PHP變量)中存儲的已知字符串長度.但是,由于strlen()是函數(shù),多多少少會有些慢,因為函數(shù)調(diào)用會經(jīng)過諸多步驟,如字母小寫化(譯注:指函數(shù)名小寫化,PHP不區(qū)分函數(shù)名大小寫)、哈希查找,會跟隨被調(diào)用的函數(shù)一起執(zhí)行.在某些情況下,你可以使用isset()?技巧加速執(zhí)行你的代碼.
(舉例如下)
PHP教程if (strlen($foo) < 5) {?echo “Foo is too short”$$ }
PHP教程(與下面的技巧做比較)
PHP教程if (!isset($foo{5})) { echo “Foo is too?short”$$ }
PHP教程調(diào)用isset()恰巧比strlen()快,因為與后者不同的是,isset()作為一種語言結(jié)構(gòu),意味著它的執(zhí)行不需要函數(shù)查找和字母小寫化.也就是說,實際上在檢驗字符串長度的頂層代碼中你沒有花太多開銷.
PHP教程40.使用++$i遞增
PHP教程
When incrementing or decrementing the value of the variable $i++ happens to be a tad slower then ++$i. This is something PHP specific and does not apply to other languages, so?don't go modifying your?C or Java code thinking?it'll suddenly become faster, it won't.?++$i happens to?be faster in PHP because instead of 4 opcodes used for $i++ you?only need 3. Post incrementation actually causes in the?creation of a temporary?var that is then incremented. While preincrementation increases?the original value directly. This is one of the?optimization that opcode optimized like?Zend's PHP optimizer. It is a still a good idea?to keep?in mind?since not all opcode optimizers?perform?this optimization and there are?plenty of ISPs and servers running without an opcode optimizer.
PHP教程當執(zhí)行變量$i的遞增或遞減時,$i++會比++$i慢一些.這種差異是PHP特有的,并不適用于其他語言,所以請不要修改你的C或Java代碼并指望它們能立即變快,沒用的.++$i更快是因為它只需要3條指令(opcodes),$i++則需要4條指令.后置遞增實際上會產(chǎn)生一個臨時變量,這個臨時變量隨后被遞增.而前置遞增直接在原值上遞增.這是最優(yōu)化處理的一種,正如Zend的PHP優(yōu)化器所作的那樣.牢記這個優(yōu)化處理不失為一個好主意,因為并不是所有的指令優(yōu)化器都會做同樣的優(yōu)化處理,并且存在大量沒有裝配指令優(yōu)化器的互聯(lián)網(wǎng)服務(wù)
提供商(ISPs)和服務(wù)器.
PHP教程40. 不要隨便就復(fù)制變量
PHP教程有時候為了使 PHP 代碼更加整潔,一些 PHP?新手(包括我)會把預(yù)定義好的變量復(fù)制到一個名字更簡短的變量中,其實這樣做的結(jié)果是增加了一倍的內(nèi)存消耗,只會使程序更加慢.試想一下,在下面的例子中,如果用戶惡意插入 512KB 字節(jié)的文字到文本輸入框中,這樣就會導(dǎo)致?1MB 的內(nèi)存被消耗!
PHP教程
BAD:
$description = $_POST['description'];
echo $description;
GOOD:
echo $_POST['description'];
PHP教程41 使用選擇分支語句
PHP教程switch case好于使用多個if,else?if語句,并且代碼更加容易閱讀和維護.
PHP教程42.在可以用file_get_contents替代file、fopen、feof、fgets
PHP教程在可以用file_get_contents替代file、fopen、feof、fgets等系列方法的情況下,盡量用file_get_contents,因為他的效率高得多!但是要注意file_get_contents在打開一個URL文件時候的PHP版本問題;
PHP教程43.盡量的少進行文件操作,雖然PHP的文件操作效率也不低的;
PHP教程44.優(yōu)化Select SQL語句,在可能的情況下盡量少的進行Insert、Update操作(在update上,我被惡批過);
PHP教程45.盡可能的使用PHP內(nèi)部函數(shù)
PHP教程46.循環(huán)內(nèi)部不要聲明變量,尤其是大變量:對象
PHP教程(這好像不只是PHP里面要注意的問題吧?);
PHP教程47.多維數(shù)組盡量不要循環(huán)嵌套賦值;
PHP教程48.foreach效率更高,盡量用foreach代替while和for循環(huán);
PHP教程49.“用i+=1代替i=i+1.符合c/c++的習(xí)慣,效率還高”;
PHP教程50.對global變量,應(yīng)該用完就unset()掉;
PHP教程51 并不是事必面向?qū)ο?OOP),面向?qū)ο笸_銷很大,每個方法和對象調(diào)用都會消耗很多內(nèi)存.
PHP教程52 不要把方法細分得過多,仔細想想你真正打算重用的是哪些代碼?
PHP教程53 如果在代碼中存在大量耗時的函數(shù),你可以考慮用C擴展的方式實現(xiàn)它們.
PHP教程54、打開apache的mod_deflate模塊,可以提高網(wǎng)頁的瀏覽速度.
(提到過echo 大變量的問題)
PHP教程55、數(shù)據(jù)庫連接當使用完畢時應(yīng)關(guān)掉,不要用長連接.
PHP教程56、split比exploade快
PHP教程
split()
0.001813 - 0.002271 seconds (avg 0.002042 seconds)
explode()
0.001678 - 0.003626 seconds (avg 0.002652 seconds)
Split can take regular expressions as delimiters, and runs faster too. ~23% on average.
轉(zhuǎn)載請注明本頁網(wǎng)址:
http://www.fzlkiss.com/jiaocheng/5898.html