[譯] 又一個(gè)升級(jí)到 PHP 7 后性能提升 50% 的案例:Tumblr
導(dǎo)讀:PHP 7 發(fā)布有一段時(shí)間了,業(yè)界有一些公司已經(jīng)嘗試在線上大規(guī)模使用這個(gè)版本了。 Tumblr 也從 PHP 5 升級(jí)到 PHP 7,他們碰到了什么問題,又是怎么處理的,PHP7 這個(gè)版本又給 Tumblr 帶來了什么?相關(guān)的經(jīng)歷對(duì)類似需求的團(tuán)隊(duì)會(huì)有很多啟發(fā) 。
在 Tumblr,我們一直在尋找新方法來提高網(wǎng)站的性能。 這意味著在關(guān)鍵路徑大量使用緩存,測(cè)試新的 CDN 配置或升級(jí)底層軟件。
最近,在團(tuán)隊(duì)的努力下,我們將整個(gè) Web 服務(wù)器從 PHP 5 升級(jí)到 PHP 7,整個(gè)升級(jí)過程非常有意思,因此通過本文分享如下。
升級(jí)的過程
這一切都開始于 2015 年秋天一個(gè) hackday 項(xiàng)目。@oli 和 @trav 在 Tumblr 運(yùn)行一個(gè) PHP 7 rc 版本。在那時(shí)候,很多 PHP 擴(kuò)展都還不支持PHP 7,但有非官方的分支保證實(shí)驗(yàn)性支持。盡管這樣,Tumblr 已經(jīng)可以運(yùn)行在 PHP7 上了!
今年春天,Tumblr 可以更加穩(wěn)定的運(yùn)行在 PHP7 上,我們決定開始升級(jí)。我們做的第一件事是打包新版本,以便更容易安裝并保證一致。與此同時(shí),我們將內(nèi)部 PHP 擴(kuò)展移植到新版本,所以一切都準(zhǔn)備好了。
我們編寫了一個(gè)可以升級(jí)(或降級(jí))服務(wù)器的小腳本。然后,在春季及夏季,一直都在運(yùn)行測(cè)試并進(jìn)行性能監(jiān)測(cè)和評(píng)估。隨著時(shí)間推移,我們開始在更多開發(fā)人員日常使用的機(jī)器里運(yùn)行PHP 7。
最后,在 8 月底,我們對(duì)測(cè)試結(jié)果感到滿意,于是將 PHP 7 替換到我們一小部分生產(chǎn)服務(wù)器。兩個(gè)星期后,所有線上服務(wù)器都已更新到 PHP7。
測(cè)試與持續(xù)集成當(dāng)進(jìn)行這樣的升級(jí)時(shí),用測(cè)試以確保代碼正確運(yùn)行是非常重要的,我們有幾個(gè)方法。
在這個(gè)項(xiàng)目中,我們使用 Phan 尋找代碼庫(kù)中與 PHP 7 不兼容的代碼。
Phan:https://github.com/etsy/phan
我們還有一套單元和集成測(cè)試,幫助我們識(shí)別有問題的模塊。 由于項(xiàng)目開發(fā)還在進(jìn)行中,我們需要確保沒有添加 PHP7 不兼容的新代碼,所以我們?cè)O(shè)置了持續(xù)集成的任務(wù),同時(shí)在 PHP5 和 PHP7 環(huán)境下運(yùn)行所有測(cè)試。
升級(jí)結(jié)果最終的結(jié)果是什么? 性能和語(yǔ)言特性對(duì)我們來說是最重要的兩個(gè)結(jié)果。
性能當(dāng)我們?cè)诘谝慌?wù)器上運(yùn)行 PHP7 時(shí),我們非常密切地關(guān)注各種性能監(jiān)控圖表,以確保運(yùn)行順利。 正如我們上面提到的,我們正在進(jìn)行性能改進(jìn),實(shí)際上的升級(jí)結(jié)果是非常驚人的, 延遲減少一半,并且服務(wù)器上的 CPU 負(fù)載降低了 50% 或者更多 。不但頁(yè)面響應(yīng)速度提高了一倍,而且只使用了一半的 CPU 資源。
這些是來自我們的API服務(wù)器的圖。 如您所見,延遲下降到不到一半,峰值負(fù)載平均值現(xiàn)在低于之前的最低點(diǎn)!
語(yǔ)言特性除此之外,PHP 7 也帶來了很多有趣的新功能。 一些亮點(diǎn)是:
標(biāo)量類型提示:PHP 歷史版本在類型安全方面相當(dāng)差,PHP 7 引入標(biāo)量類型提示( Scalar type ),確保傳遞的值符合特定類型(字符串,bool,int,float等)。
返回類型聲明:現(xiàn)在,使用 PHP 7,函數(shù)可以有顯式返回類型。 這減少了對(duì)樣板代碼和手動(dòng)檢查函數(shù)返回值的需求。
匿名類:與匿名函數(shù)(閉包)非常類似,匿名類在運(yùn)行時(shí)構(gòu)建,可以模擬類/接口甚至擴(kuò)展其他類。 這些對(duì)于像日志類這樣的實(shí)用對(duì)象非常有用,并且在單元測(cè)試中非常有用。
各種安全和性能增強(qiáng)。
總結(jié)PHP7 相當(dāng)棒。
以下是國(guó)外網(wǎng)站一些網(wǎng)友評(píng)論。
jsjohnst :
我沒有參與這個(gè)升級(jí)過程,但在基于我對(duì) Tumblr(過去在那里工作了 3 年)代碼庫(kù)的了解,以及個(gè)人對(duì) PHP 的背景(一直是 PHP 貢獻(xiàn)者?13 年),性能提升有 95% 的可能性與 PHP7 相關(guān)。 Chorus (Tumblr 內(nèi)部基于的框架) 中使用的許多模式和在代碼庫(kù),都會(huì)在 PHP7 內(nèi)部的變化中受益匪淺。
tyingq:
從 PHP5.x 切換到 PHP7 時(shí),在響應(yīng)時(shí)間,TPS 和內(nèi)存使用方面的顯著改進(jìn)
https://www.symfony.fi/entry/symfony-benchmarks-php-56-hhvm-and-php-7
uses:
我們使用的 PaaS 云平臺(tái)最近升級(jí)了一個(gè) PHP 7。
對(duì)于運(yùn)行 Bolt 的網(wǎng)站(一個(gè)PHP CMS,它大量使用自動(dòng)加載類),我們看到性能提高了 45% ~ 55%。 對(duì)于 Wordpress,它是約 30 ~ 40%。
bkanber :
我覺得也可能是因?yàn)?PHP7 默認(rèn)啟用了 opcache,而之前并沒有啟用。
也就是說,更快的解析器和 intepreter 也會(huì)做更快的自動(dòng)加載,所以即使不是opcache,也可能是憑借自動(dòng)加載類被解釋得更快。
nikic:
我們?cè)?7.1 版中 perf 顯示在實(shí)際應(yīng)用中沒有看到顯著的收益(對(duì)計(jì)算密集型代碼有一些改進(jìn),比如循環(huán)中的計(jì)算)。
JIT 是否會(huì)對(duì)應(yīng)用程序帶來重大改進(jìn)尚不清楚,我們希望如此。
boulos:
對(duì),我很久以前也對(duì)比過,大多數(shù)的性能改進(jìn)來自于運(yùn)行時(shí)的 C 代碼改進(jìn),比如在實(shí)際運(yùn)行的系統(tǒng)中,產(chǎn)生較少的字符串副本非常關(guān)鍵。這是你在 PHP 7 改進(jìn)中首先看到的。
使用 PHP 從字節(jié)碼解釋器生成 x86 機(jī)器碼將意味著在你不結(jié)束“跳出”的地方,你應(yīng)該看到更接近 C 代碼性能(JavaScript,Python 等同樣如此)。PHP7 的標(biāo)量類型提示和其他一些小調(diào)整(有意義的)應(yīng)該使 jit 編程一個(gè)更容易的任務(wù)(對(duì)比從 PHP 5.4 到 PHP7)。類型推理是非常有效的,在實(shí)踐中(見Paul Biggar對(duì)此的優(yōu)秀論文),在 PHP 程序中確實(shí)沒有那么多的多態(tài)性。函數(shù)的參數(shù)通常有一個(gè)或兩個(gè)類型,或者它們實(shí)際上是多態(tài)的并有跟多的變體。
progforlyfe
沒錯(cuò),最大的性能改進(jìn)是來自類似 JIT 改進(jìn), 跟 HHVM 所做的工作類似。
tiffanyh:
我很困惑,PHP 7 現(xiàn)在沒有 JIT,但它的性能已經(jīng)與 HHVM(它有JIT)一致。
然后就是說 PHP 8(與JIT)將顯著快于 HHVM?
TazeTSchnitzel :
未必,這可能意味著很多 HHVM 的性能改進(jìn)與 JIT 無(wú)關(guān)。
來自:http://mp.weixin.qq.com/s?__biz=MzAwMDU1MTE1OQ==&mid=2653547882&idx=1&sn=12512269ddb4743c6c7978277643d28b&chksm=813a7cf2b64df5e406ad8ebee973da19d330b3785a134af577883e5f725fa28884d070ed9cf7&mpshare=1&scene=1&srcid=1115vhT3jmCZQ5qd7jtLKtf6&from=groupmessage&isappinstalled=0
相關(guān)文章:
1. 一個(gè) 2 年 Android 開發(fā)者的 18 條忠告2. Vue實(shí)現(xiàn)仿iPhone懸浮球的示例代碼3. js select支持手動(dòng)輸入功能實(shí)現(xiàn)代碼4. vue-drag-chart 拖動(dòng)/縮放圖表組件的實(shí)例代碼5. 什么是Python變量作用域6. Spring的異常重試框架Spring Retry簡(jiǎn)單配置操作7. Android 實(shí)現(xiàn)徹底退出自己APP 并殺掉所有相關(guān)的進(jìn)程8. PHP正則表達(dá)式函數(shù)preg_replace用法實(shí)例分析9. Android studio 解決logcat無(wú)過濾工具欄的操作10. vue使用moment如何將時(shí)間戳轉(zhuǎn)為標(biāo)準(zhǔn)日期時(shí)間格式
