被swoole坑哭的PHP程序員
本文主要記錄一下學(xué)習(xí)swoole的過(guò)程、填過(guò)的坑以及swoole究竟有多么強(qiáng)大!
首先說(shuō)一下對(duì)swoole的理解:披著PHP外衣的C程序。很多PHPer朋友看到swoole提供的強(qiáng)大功能、外界對(duì)其的崇拜便躍躍欲試的安裝、調(diào) 試其demo、編寫(xiě)新功能,然后興奮的奔走相告。沒(méi)過(guò)幾天當(dāng)你按照自己的理解繼續(xù)用swoole時(shí),發(fā)現(xiàn)代碼并沒(méi)有按照自己的預(yù)期運(yùn)行,然后開(kāi)始破口大 罵,什么破東西呀,代碼跟demo基本一樣,為啥運(yùn)行不通呢?什么狗屁work、task、共享內(nèi)存、ipcs、異步,各種問(wèn)題涌現(xiàn),然后迅速去查官方文 檔,發(fā)現(xiàn)文檔中竟然對(duì)這些并沒(méi)有提及,只是簡(jiǎn)單的介紹怎么使用,此時(shí)幾乎對(duì)swoole喪失希望。
遇到的幾點(diǎn)問(wèn)題:
1:關(guān)于phper常用的全局變量(global)為什么在onRequest函數(shù)中不能使用。
因?yàn)閟woole是多線(xiàn)程編程,global是不能在多個(gè)進(jìn)程間共享的。例
- global $i = 0;
- function onRequest() {
- echo $i++;
- }
如果在swoole中寫(xiě)一個(gè)上面的程序,并不會(huì)每次訪(fǎng)問(wèn)輸出一個(gè)遞增的數(shù)字。如果要實(shí)現(xiàn)預(yù)期的效果,需要使用swoole_table的相關(guān)函數(shù)。
2:什么是異步、什么是回高
對(duì)于phper來(lái)說(shuō),對(duì)異步、回調(diào)的理解估計(jì)就是ajax。當(dāng)看到swoole里面對(duì)異步、回調(diào)的解釋?zhuān)菜坪芎?jiǎn)單的樣子,就這樣在沒(méi)有任何多線(xiàn)程編輯經(jīng)驗(yàn)的時(shí)候貿(mào)然用了swoole,結(jié)果被坑的偷偷擼代碼好幾個(gè)通宵來(lái)填自己的坑。
3:為什么onReceive收到的數(shù)據(jù)這么大
客戶(hù)端發(fā)送的多次請(qǐng)求,服務(wù)端是可以一次性接收的。并不是客戶(hù)端發(fā)送一次,服務(wù)端接收一次
4:自制httpserve
寫(xiě)一個(gè)http服務(wù)端,然后通過(guò)瀏覽器訪(fǎng)問(wèn)這個(gè)自制的服務(wù)器,刷新一次瀏覽器,服務(wù)端為什么為接收到兩次請(qǐng)求?這個(gè)問(wèn)題估計(jì)困饒了好多初次用swoole寫(xiě)httpserver的朋友。因?yàn)闉g覽器會(huì)多發(fā)一個(gè)favicon.ico請(qǐng)求。
原因
出現(xiàn)這種情況的原因其實(shí)很簡(jiǎn) 單,大部分phper都只會(huì)php這一種語(yǔ)言,主要用途就是做web,寫(xiě)業(yè)務(wù)邏輯。很少去了解服務(wù)器程序的開(kāi)發(fā)。有一次一個(gè)朋友用swoole寫(xiě)了一個(gè)簡(jiǎn) 單的服務(wù)端,一個(gè)客戶(hù)端,跑過(guò)來(lái)問(wèn)我為什么都啟動(dòng)了卻都收不到數(shù)據(jù),我簡(jiǎn)單看了下代碼,所有連接確實(shí)都成功了,兩端都設(shè)置了onReceive回調(diào),代碼 沒(méi)問(wèn)題,看到最后才發(fā)現(xiàn)他的服務(wù)端、客戶(hù)端都設(shè)置了接到消息的回調(diào)函數(shù),但是兩端都沒(méi)有向?qū)Ψ桨l(fā)消息,兩端處于僵持狀態(tài)。然后swoole官方對(duì)于這種常 識(shí)問(wèn)題沒(méi)有給出說(shuō)明,只是說(shuō)如何設(shè)置回調(diào)、如何發(fā)消息,如何這樣,如何那樣。對(duì)于有服務(wù)端開(kāi)發(fā)經(jīng)驗(yàn)的同學(xué)來(lái)說(shuō),肯定不會(huì)遇到這種問(wèn)題,swoole文檔也 不需要指明需要這樣做,因?yàn)檫@是常識(shí)。但對(duì)于phper來(lái)說(shuō),指明這一點(diǎn)是非常重要的,因?yàn)槿缟厦嫠f(shuō)phper是沒(méi)有這方面認(rèn)知的,只有服務(wù)端開(kāi)發(fā)經(jīng)驗(yàn) 的程序員有才會(huì)有。
swoole的特色:網(wǎng)絡(luò)通信 框架、異步、多線(xiàn)程。這些特性正是php所不完善的功能(雖然官方提供很多基礎(chǔ)函數(shù)可以實(shí)現(xiàn)這些功能,然后缺少中文文檔,很少有人用php來(lái)實(shí)現(xiàn)這部分功 能),普通的phper也不具備這些特性的基礎(chǔ)認(rèn)知,所以貿(mào)然使用swoole難免會(huì)遇到一些根本在swoole官方查不到的常識(shí)問(wèn)題。
使用swoole必須要掌握的技能
-
多線(xiàn)程編程
-
進(jìn)程間通信
-
網(wǎng)絡(luò)協(xié)議TCP/UDP的認(rèn)知
-
PHP的各項(xiàng)基本技能
個(gè)人學(xué)習(xí)swoole的經(jīng)歷
在很久之前我也是一個(gè)只會(huì) php的程序員,后來(lái)一次偶然機(jī)會(huì)需要用httpsqs,用了一段時(shí)間后發(fā)現(xiàn)有一些個(gè)性的需求,于是就開(kāi)始看源碼。這真是不看不知道,一看嚇一 跳,httpsqs只是一層簡(jiǎn)單的包裝,內(nèi)部是一個(gè)Tokyo Cabinet數(shù)據(jù)庫(kù),印象中封裝的代碼也就一百多行。主要思路就是用C語(yǔ)言的libevent做了一個(gè)http服務(wù)器,接收請(qǐng)求讀寫(xiě)tokyo cabinet數(shù)據(jù)庫(kù),當(dāng)時(shí)按照這種思路做出來(lái)的程序確實(shí)不少。后來(lái)我就突發(fā)奇想,既然C語(yǔ)言可以用libevent函數(shù),那PHP肯定也可以用 libevent監(jiān)聽(tīng)網(wǎng)絡(luò),接收請(qǐng)求后讀寫(xiě)數(shù)據(jù)庫(kù)做隊(duì)列服務(wù)。后來(lái)經(jīng)過(guò)查php官方文檔,PHP確實(shí)提供一系統(tǒng)完整的函數(shù)來(lái)完成這些功能,甚至多線(xiàn)程的全 套函數(shù)都有提供,但中文文檔太少,網(wǎng)上也很少搜索到成熟的代碼。在逼不得已的情況下,補(bǔ)習(xí)了linux-C多線(xiàn)程開(kāi)發(fā)的基本原理,進(jìn)程間通信的常用方法, 也用來(lái)做了一些簡(jiǎn)單的demo。唯一的感覺(jué)就是寫(xiě)一個(gè)簡(jiǎn)單的功能,設(shè)計(jì)起來(lái)還真復(fù)雜。就在快要放棄的時(shí)候,swoole出現(xiàn)了。swoole所提供的功能 正是php所缺失的功能,簡(jiǎn)直是太棒了。swoole做為一種網(wǎng)絡(luò)通信框架,只需要簡(jiǎn)單的幾行設(shè)置,一個(gè)服務(wù)器就搭建起來(lái)了,以后就是不斷的去完善業(yè)務(wù)代 碼。之前在libevent交流群中得知swoole的設(shè)計(jì)在c\c++中并不是最好的框架設(shè)計(jì),但其亮點(diǎn)就是把基本功能用C封裝好,業(yè)務(wù)功能留給世界上 最好的語(yǔ)言PHP來(lái)編寫(xiě)。自此便開(kāi)始了swoole的填坑之旅。
總結(jié)
swoole并不是一個(gè)簡(jiǎn)單的PHP框架,正如swoole官方首頁(yè)的第一句話(huà)“重新定義PHP”,千萬(wàn)不要用舊有php的思想來(lái)寫(xiě)swoole代碼!swoole重新激活了PHP,php成就了swoole!