php長串連,奏是這麼簡單_PHP教程

來源:互聯網
上載者:User

php長串連,奏是這麼簡單


說到長連結大家肯定不陌生,就是複用一個連結持續不斷的進行資料互動,它不像那些一夜情似的服務,需要頻繁的開啟和關閉連結,效率低的同時還增加了業務的複雜度。在襠下很多互連網業務情境都需要長串連的支援,比如:遊戲、聊天、資訊推送等等等,今天我們就一步一步來揭秘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....

 
  1. $sfd = socket_create(AF_INET, SOCK_STREAM, 0);
  2. socket_bind($sfd, "0.0.0.0", 1234);
  3. socket_listen($sfd, 511);
  4. socket_set_option($sfd, SOL_SOCKET, SO_REUSEADDR, 1);
  5. socket_set_nonblock($sfd);
  6. $rfds = array($sfd);
  7. $wfds = array();
  8. do{
  9. $rs = $rfds;
  10. $ws = $wfds;
  11. $es = array();
  12. $ret = socket_select($rs, $ws, $es, 3);
  13. //read event
  14. foreach($rs as $fd){
  15. if($fd == $sfd){
  16. $cfd = socket_accept($sfd);
  17. socket_set_nonblock($cfd);
  18. $rfds[] = $cfd;
  19. echo "new client coming, fd=$cfd\n";
  20. }else{
  21. $msg = socket_read($fd, 1024);
  22. if($msg <= 0){
  23. //close
  24. }else{
  25. //recv msg
  26. echo "on message, fd=$fd data=$msg\n";
  27. }
  28. }
  29. }
  30. //write event
  31. foreach($ws as $fd){
  32. socket_write($fd, ........);
  33. }
  34. }while(true);

0x02 登峰造極

從玩socket的那天起,google就輕言細語的跟我說,高並發下的select不要用啊,效率底啊,win要用iocp啊, linux要用epoll啊,blablablabla...哦!好吧,既然google都這麼說了,我也不能跟他老人家較真不是,又一次決定(為什麼要說又呢?)要聽google話,把epoll搞起來,可總不能自己寫啊?像我這麼懶的人還是整個擴充好了,libevent走你!經過瘋狂的編(co)碼(py),神作終於出山,具體能有多高效,能撐多少並發,不造,反正沒用select了,我奏是屌!

 
  1. $sfd = stream_socket_server ('tcp://0.0.0.0:1234', $errno, $errstr);
  2. stream_set_blocking($sfd, 0);
  3. $base = event_base_new();
  4. $event = event_new();
  5. event_set($event, $sfd, EV_READ | EV_PERSIST, 'ev_accept', $base);
  6. event_base_set($event, $base);
  7. event_add($event);
  8. event_base_loop($base);
  9. function ev_accept($socket, $flag, $base)
  10. {
  11. $connection = stream_socket_accept($socket);
  12. stream_set_blocking($connection, 0);
  13. $buffer = event_buffer_new($connection, 'ev_read', NULL, 'ev_error', $connection);
  14. event_buffer_base_set($buffer, $base);
  15. event_buffer_timeout_set($buffer, 30, 30);
  16. event_buffer_watermark_set($buffer, EV_READ, 0, 0xffffff);
  17. event_buffer_priority_set($buffer, 10);
  18. event_buffer_enable($buffer, EV_READ | EV_PERSIST);
  19. }
  20. function ev_error($buffer, $error, $connection)
  21. {
  22. event_buffer_disable($buffer, EV_READ | EV_WRITE);
  23. event_buffer_free($buffer);
  24. fclose($connection);
  25. }
  26. function ev_read($buffer, $connection)
  27. {
  28. $read = event_buffer_read($buffer, 256);
  29. //do something....
  30. }

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..

 
  1. $sid = $_SERVER['SESSIONID'];//這是sessionid
  2. $data = file_get_contents("php://input");//這樣就能拿到請求內容了
  3. //想要返回訊息只需要兩步
  4. header('Content-Length: 11');//返回給用戶端位元組數
  5. echo "hello world";
  6. //想要給別的使用者發訊息
  7. include 'api.php';
  8. $router = new RouterClient('router host', 'router port');
  9. $router->sendMsg(使用者sessionid, "fuck you");
  10. //想要給所有人要訊息
  11. $router->sendAllMsg("fuck all");
  12. //想給指定組發訊息(類似redis的pub/sub)
  13. $router->publish("channel name", "fuck all");

項目地址: http://git.oschina.net/scgywx/fooking

文檔地址(不定期更新):http://my.oschina.net/scgywx/blog/465186

http://www.bkjia.com/PHPjc/1015229.htmlwww.bkjia.comtruehttp://www.bkjia.com/PHPjc/1015229.htmlTechArticlephp長串連,奏是這麼簡單 說到長連結大家肯定不陌生,就是複用一個連結持續不斷的進行資料互動,它不像那些一夜情似的服務,需要頻繁...

  • 聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

    如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

    A Free Trial That Lets You Build Big!

    Start building with 50+ products and up to 12 months usage for Elastic Compute Service

    • Sales Support

      1 on 1 presale consultation

    • After-Sales Support

      24/7 Technical Support 6 Free Tickets per Quarter Faster Response

    • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.