別再像2009年那樣寫PHP代碼了
離開在Facebook擔(dān)任工程師的僅僅2個(gè)月時(shí)間,我就很困惑,外面的世界看上去仍然像是在2009年的時(shí)候那樣寫 PHP。
貌似人們從來(lái)沒(méi)聽過(guò) Hack、 HHVM、 XHP 等等,人們?nèi)耘f在代碼里大量使用 require() 和 include() 語(yǔ)句。簡(jiǎn)直了。
我仍然認(rèn)為 PHP 是一門寫前端應(yīng)用的優(yōu)秀語(yǔ)言(業(yè)務(wù)邏輯和 API 層),但只有當(dāng)你應(yīng)用了以下它的現(xiàn)代優(yōu)勢(shì)時(shí),這一說(shuō)法才成立:
1. Hack
打出你的變量:
說(shuō)實(shí)話,PHP 最大的問(wèn)題是它缺乏強(qiáng)類型。 變量可以是任何類型,很多時(shí)候這就是一個(gè)定時(shí)炸彈。
如果你不得不寫這樣的代碼:
- if ($var !== null && is_int($var)) {
- //...
- }
這意味著你可以想引用一個(gè)null變量,或者錯(cuò)誤的變量類型。
Hack 是向 PHP 漸漸添加類型信息的途徑,而且它是基于 PHP 的。
如果你添加了 hack 類型提示,它強(qiáng)制約束你的變量(包括把他們標(biāo)記為可能為 null)。例如:
- class Foo {
- ?int $var = null; // ... some code ...
- }
可以用在方法簽名、類屬性等上面,接著它允許你通過(guò) hh_client 檢查代碼里是否存在錯(cuò)誤,存在就會(huì)把類型錯(cuò)誤高亮出來(lái)。
Hack 文檔頁(yè)面有更多更好的對(duì)于 Hack 類型的解釋: https://docs.hhvm.com/hack/overview/typing
Async 異步
對(duì)于體面的 PHP 網(wǎng)站來(lái)說(shuō),下一個(gè)重要的跨越是使用 hack 的 async/await 關(guān)鍵詞。
如果你從未接觸過(guò)類似特性的語(yǔ)言,我來(lái)解釋一下。
比如講,你需要對(duì)數(shù)據(jù)庫(kù)做 3 次函數(shù)調(diào)用,為了獲取 3 塊數(shù)據(jù)。為了計(jì)算出頁(yè)面想要的結(jié)果,你需要所有 3 個(gè)查詢結(jié)果,但每個(gè)結(jié)果都需要 1 條不一樣的 SQL 語(yǔ)句。
一般你會(huì)這樣寫:
- $data1 = querySQL1();
- $data2 = querySQL2();
- $data3 = querySQL3();
- $result = computeResult($data1, $data2, $data3);
好,實(shí)際上,除非你在明確的做一些牛逼的東西,PHP 通常是在一個(gè)請(qǐng)求里面單線程跑的。 這意味著服務(wù)器會(huì)首先給第一條查詢執(zhí)行一條 SQL,等待結(jié)果,然后再執(zhí)行第二條 SQL,接著再執(zhí)行第三條。
這有什么問(wèn)題呢?這里的問(wèn)題是,計(jì)算最終結(jié)果所需的時(shí)間是執(zhí)行 query1、query2 和 query3 三者的時(shí)間之和。
但大多數(shù)數(shù)據(jù)庫(kù)都是多線程,且可以并行執(zhí)行操作的。如果在此之上,你的 DB 在 SSD 上而不是在機(jī)械硬盤上執(zhí)行,你就可以利用上 DB 的多核處理器和并行處理能力...
如果你在查詢多個(gè) DB 或者多個(gè)不同的服務(wù),或是請(qǐng)求多個(gè) API,對(duì)你來(lái)說(shuō)這一特性也可以發(fā)揮優(yōu)勢(shì)。
我們?cè)趺磥?lái)解決呢? 使用 async/await:
- list($data1, $data2, $data3) = await\HH\Asio\v(array(
- querySQL1(),
- querySQL2(),
- querySQL3(),
- ));
是這種方式,3 條查詢一次性發(fā)送并等待結(jié)果?,F(xiàn)在獲取 3 塊數(shù)據(jù)的時(shí)間就是執(zhí)行耗時(shí)最長(zhǎng)那條查詢的時(shí)間,因?yàn)?3 條都在并行處理。
Hack 使用圖表的方式更好地對(duì) async 做了解釋:https://docs.hhvm.com/hack/async/introduction
Hack 提供了對(duì) MySQL, memcache 和 Curl 的 async 實(shí)現(xiàn),所以你可以只需用它們的庫(kù)替換掉你的調(diào)用就能立即利用到這一優(yōu)勢(shì)。
Collections:
PHP 數(shù)據(jù),有時(shí)候是一個(gè)向量,有時(shí)候是一個(gè)字典,有時(shí)候兩者都是。
即便你知道它里面包括什么,其他的工程師很可能認(rèn)為自己也知道,但卻在里面放進(jìn)了錯(cuò)誤的數(shù)據(jù)類型。
如果你曾經(jīng)使用過(guò)像 C#, Java 或 C++ 這樣的語(yǔ)言,你可能對(duì) Generics 和 Collections 會(huì)感到熟悉。
Hack 引入了 Collections, 它讓你指定 Collections 里面的數(shù)據(jù)類型。 這意味著你只是盲目寄望于數(shù)組包含了你想要的值,現(xiàn)在你知道這一結(jié)構(gòu)包括了你想要的數(shù)據(jù)類型(字符串、整型等等)。
在這之上,如果你仍舊想使用 PHP 的數(shù)組,你只需要對(duì)代碼做一點(diǎn)點(diǎn)重構(gòu),你就可以對(duì)數(shù)組內(nèi)容的類型進(jìn)行這樣的約束:
- class Bar { array
- $vector_of_ints = array();
- array
- $dictionary_with_string_keys = array();
- }
然后你只要在數(shù)組里放置了錯(cuò)誤類型的變量,或者給數(shù)組指定一個(gè)字符串鍵,類型檢查器就會(huì)拋出錯(cuò)誤。
2. HHVM
Hack 帶有它自己的運(yùn)行環(huán)境,如你預(yù)料的,它無(wú)法直接運(yùn)行于 Zend 的 PHP 環(huán)境。
HHVM 指 HipHop 虛擬機(jī),是在 Facebook 開發(fā)的旨在極大改進(jìn) PHP 規(guī)模化的執(zhí)行復(fù)雜度問(wèn)題。
HHVM 運(yùn)行了整個(gè) Facebook 和一些其他主要站點(diǎn),比如現(xiàn)在的維基百科,隨著時(shí)間推移,越來(lái)越證明它所帶來(lái)的許多性能收益。
由于 HHVM 無(wú)需 Hack 提示符也可以運(yùn)行常規(guī)的 PHP,且同樣可以加速代碼執(zhí)行效率,所以不使用 HHVM 作為你默認(rèn)的 PHP 運(yùn)行環(huán)境就是在浪費(fèi)錢。
例如,當(dāng)維基百科切換至 HHVM 后,平均單頁(yè)加載時(shí)間減少了超過(guò)一半,CPU 的平均使用率從 70% 減少至 12%,這還是在 2 年前。自那時(shí)起, HHVM 團(tuán)隊(duì)持續(xù)提升其性能表現(xiàn),所以你可以想象它現(xiàn)在表現(xiàn)更好了。
HHVM 在生產(chǎn)環(huán)境需要一個(gè)像 Apache 或 nginx 這樣的 HTTP 服務(wù)器作為前端支撐,但是在開發(fā)環(huán)境,它也可以獨(dú)立作為服務(wù)器運(yùn)行。
3. XHP
如果有一件事是我憎惡的,就是 PHP/HTML 混編。這樣的代碼讓我吐:
- $user_name = 'Fred';
- $output = "Hello $user_name";
更早的是,有人自作聰明,不在一個(gè)地方開閉 HTML 標(biāo)簽,像這樣:
- $user_name = 'Fred'; $output = "
- Hello $user_name"; // some call to a function that takes in $output and is supposed to close the div tag $output = addTheRestOfTheSoup($output);
于是你維護(hù)起來(lái)就...
XHP 讓 HTML 作為 PHP 的一級(jí)公民,因此你可以在字符串外編寫 HTML,像 XHP 一樣解析。
比如:
- $user_name ='Fred'; $output =
- Hello $user_name; addTheRestOfTheDivContentsTo($output);
- //...
- function addTheRestOfTheDivContentsTo(:div $div): :div { $div->appendChild("We come in peace"); return $div; }
如你所見, XHP 同樣強(qiáng)制標(biāo)簽匹配,也就是說(shuō)開標(biāo)簽有相應(yīng)的閉標(biāo)簽,且以合適的順序進(jìn)行開閉。
XHP 同樣處理字符串變量的 escape,避免 HTML/JS 進(jìn)入頁(yè)面的用戶內(nèi)容中,防御網(wǎng)站受到該攻擊矢量的攻擊。
你還可以為你自己創(chuàng)建自定義的 XHP 類,比如“自定義的HTML標(biāo)簽”來(lái)復(fù)用你的代碼庫(kù),比如實(shí)現(xiàn)可以自動(dòng)在 Facebook 頁(yè)面添加鏈接的功能,甚至用一個(gè)標(biāo)簽來(lái)渲染整個(gè)頁(yè)面頭部。
更多關(guān)于 XHP 的文檔:https://docs.hhvm.com/hack/XHP/introduction
還有更多 ...
以上介紹了 HHVM、Hack 和 XHP 的基礎(chǔ),下次我希望介紹一下設(shè)置 HHVM 的開發(fā)環(huán)境,基于 HHVM 的類自動(dòng)加載、函數(shù)和常量,還有基礎(chǔ)的控制器框架,路由 web 請(qǐng)求。