說到長連結大家肯定不陌生,就是複用一個連結持續不斷的進行資料互動,它不像那些一夜情似的服務,需要頻繁的開啟和關閉連結,效率低的同時還增加了業務的複雜度。在襠下很多互連網業務情境都需要長串連的支援,比如:遊戲、聊天、資訊推送等等等,今天我們就一步一步來揭秘php長串連的玩法。我相信任何一項技術的實施都是因為業務情境的需要,所以這次我們還拿聊天室說事兒。
0x00 初試牛刀
記得以前用php寫聊天室還是用polling的方式,毫無疑問,一提到polling,肯定會有人說long polling,沒錯!long polling也很不錯,但在nginx+fpm上面玩這個多少有些費勁,畢竟一個請求需要佔一個php進程(就算是用apache+php_mod,也需要一個請求一個線程),所以要是幾個人隨便玩玩還行,一旦放到線上人多起來,這基本就廢了。所以還是採用polling的方式,這樣不會阻塞進程,並且一個請求能立即得到響應,但是帶來的新問題是需要不停的向伺服器發送請求,而且隨著間隔的時間越大導致訊息延遲就越大。
0x01 華麗變身
在經曆了上面那種一秒一小卡,三秒一大卡的場面!再也看不下去了,於是決定變身為真正的男人,哦不對,應該是真正的長串連。去他媽的polling, 去他媽的long polling,去他媽的webserver,統統靠邊站,讓flash socket(或者說websocket)來統治這個世界!開始了真正意義上的長串連之旅。要玩長串連總是少不了跟socket打交道吧,作為世界上最好的語言(沒有之一),socket的封裝自然是少不了滴。抄起socket_***就開幹,於是就有了下面這一托代碼,長串連是吧?延遲是吧?socket是吧?湯藥費是吧?so easy....
- $sfd = socket_create(AF_INET, SOCK_STREAM, 0);
-
- socket_bind($sfd, "0.0.0.0", 1234);
-
- socket_listen($sfd, 511);
-
- socket_set_option($sfd, SOL_SOCKET, SO_REUSEADDR, 1);
-
- socket_set_nonblock($sfd);
-
- $rfds = array($sfd);
-
- $wfds = array();
-
- do{
-
- $rs = $rfds;
-
- $ws = $wfds;
-
- $es = array();
-
- $ret = socket_select($rs, $ws, $es, 3);
-
-
-
- //read event
-
- foreach($rs as $fd){
-
- if($fd == $sfd){
-
- $cfd = socket_accept($sfd);
-
- socket_set_nonblock($cfd);
-
- $rfds[] = $cfd;
-
- echo "new client coming, fd=$cfd\n";
-
- }else{
-
- $msg = socket_read($fd, 1024);
-
- if($msg <= 0){
-
- //close
-
- }else{
-
- //recv msg
-
- echo "on message, fd=$fd data=$msg\n";
-
- }
-
- }
-
- }
-
-
-
- //write event
-
- foreach($ws as $fd){
-
- socket_write($fd, ........);
-
- }
-
-
-
- }while(true);
0x02 登峰造極
從玩socket的那天起,google就輕言細語的跟我說,高並發下的select不要用啊,效率底啊,win要用iocp啊, linux要用epoll啊,blablablabla...哦!好吧,既然google都這麼說了,我也不能跟他老人家較真不是,又一次決定(為什麼要說又呢?)要聽google話,把epoll搞起來,可總不能自己寫啊?像我這麼懶的人還是整個擴充好了,libevent走你!經過瘋狂的編(co)碼(py),神作終於出山,具體能有多高效,能撐多少並發,不造,反正沒用select了,我奏是屌!
- $sfd = stream_socket_server ('tcp://0.0.0.0:1234', $errno, $errstr);
-
- stream_set_blocking($sfd, 0);
-
- $base = event_base_new();
-
- $event = event_new();
-
- event_set($event, $sfd, EV_READ EV_PERSIST, 'ev_accept', $base);
-
- event_base_set($event, $base);
-
- event_add($event);
-
- event_base_loop($base);
-
- function ev_accept($socket, $flag, $base)
-
- {
-
- $connection = stream_socket_accept($socket);
-
- stream_set_blocking($connection, 0);
-
- $buffer = event_buffer_new($connection, 'ev_read', NULL, 'ev_error', $connection);
-
- event_buffer_base_set($buffer, $base);
-
- event_buffer_timeout_set($buffer, 30, 30);
-
- event_buffer_watermark_set($buffer, EV_READ, 0, 0xffffff);
-
- event_buffer_priority_set($buffer, 10);
-
- event_buffer_enable($buffer, EV_READ EV_PERSIST);
-
- }
-
- function ev_error($buffer, $error, $connection)
-
- {
-
- event_buffer_disable($buffer, EV_READ EV_WRITE);
-
- event_buffer_free($buffer);
-
- fclose($connection);
-
- }
-
- function ev_read($buffer, $connection)
-
- {
-
- $read = event_buffer_read($buffer, 256);
-
- //do something....
-
- }
0x03 絕處逢生
隨著人數的增長,並發的提升,單個進程已經滿足不了需求了,田伯光的故事告訴我們,單挑是鬥不過群P的,咋整?俗話說,大事化小,小事化,停!!別化了,再化就沒了。拆吧,把單進程拆成多進程,可是拆完之後又面臨新的問題,處理序間通訊、負載平衡、session唯一等。既然已經提出這樣的問題,肯定是有解決方案,現成的就有擴充和庫來解決這個事,比如:swoole,workerman等?相比之下swoole更屌一些,性、功能,呃!好像這樣簡寫不太雅觀,好吧,效能和功能更屌一些(桶哥,請原諒我的無聊~)。。。。等一下!!!但是,我們在使用php來開發web的時候,也沒有使用webserver相關的庫來做開發對不對?咱只是簡單的echo而已。這些繁雜的事都交給了nginx或者是apache,是他們義無反顧的頂在前面,讓我們可以專心寫邏輯。寫web我們只需要簡單的配置nginx和fpm就好了,那寫socket服務呢?我們為什麼不能像nginx+fpm一樣簡單配置就好了呢??當然能,必須能。。。。。看這個劇情怕是廣告要來了。。。
0x04 出其不意
寫socket服務不比寫web進階,都是打碼,都是完成需求,通訊那層都是固定的,只不過一個由nginx完成,另一個由自己完成。。可是現在不需要自己完成了,類似nginx+fpm的方案,fooking+fpm=php長串連,gateway用於承載串連,router用於轉寄訊息,處理序間通訊?負載平衡?session唯一?so easy..
- $sid = $_SERVER['SESSIONID'];//這是sessionid
-
- $data = file_get_contents("php://input");//這樣就能拿到請求內容了
-
- //想要返回訊息只需要兩步
-
- header('Content-Length: 11');//返回給用戶端位元組數
-
- echo "hello world";
-
- //想要給別的使用者發訊息
-
- include 'api.php';
-
- $router = new RouterClient('router host', 'router port');
-
- $router->sendMsg(使用者sessionid, "fuck you");
-
- //想要給所有人要訊息
-
- $router->sendAllMsg("fuck all");
-
- //想給指定組發訊息(類似redis的pub/sub)
-
- $router->publish("channel name", "fuck all");
項目地址: http://git.oschina.net/scgywx/fooking
文檔地址(不定期更新):http://my.oschina.net/scgywx/blog/465186