這篇文章給大家分享的是有關(guān)PHP長(zhǎng)連接實(shí)現(xiàn)與使用方法的示例分析的內(nèi)容。小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧。
創(chuàng)新互聯(lián)專(zhuān)注于企業(yè)營(yíng)銷(xiāo)型網(wǎng)站建設(shè)、網(wǎng)站重做改版、大渡口網(wǎng)站定制設(shè)計(jì)、自適應(yīng)品牌網(wǎng)站建設(shè)、H5場(chǎng)景定制、成都商城網(wǎng)站開(kāi)發(fā)、集團(tuán)公司官網(wǎng)建設(shè)、外貿(mào)網(wǎng)站制作、高端網(wǎng)站制作、響應(yīng)式網(wǎng)頁(yè)設(shè)計(jì)等建站業(yè)務(wù),價(jià)格優(yōu)惠性?xún)r(jià)比高,為大渡口等各大城市提供網(wǎng)站開(kāi)發(fā)制作服務(wù)。本文實(shí)例講述了PHP長(zhǎng)連接實(shí)現(xiàn)與使用方法。分享給大家供大家參考,具體如下:
長(zhǎng)連接技術(shù)(Long Polling)
在服務(wù)器端hold住一個(gè)連接, 不立即返回, 直到有數(shù)據(jù)才返回, 這就是長(zhǎng)連接技術(shù)的原理
長(zhǎng)連接技術(shù)的關(guān)鍵在于hold住一個(gè)HTTP請(qǐng)求, 直到有新數(shù)據(jù)時(shí)才響應(yīng)請(qǐng)求, 然后客戶(hù)端再次自動(dòng)發(fā)起長(zhǎng)連接請(qǐng)求.
那怎么樣hold住一個(gè)請(qǐng)求呢?服務(wù)器端的代碼可能看起來(lái)像這樣的
set_time_limit(0); //這句很重要, 不至于運(yùn)行超時(shí) while (true) { if (hasNewMessage()) { echo json_encode(getNewMessage()); break; } usleep(100000); //避免太過(guò)頻繁的查詢(xún) }
沒(méi)錯(cuò),就是通過(guò)循環(huán)來(lái)實(shí)現(xiàn)hold住一個(gè)請(qǐng)求, 不至于立即返回. 查詢(xún)到有新數(shù)據(jù)之后才響應(yīng)請(qǐng)求. 然后客戶(hù)端處理數(shù)據(jù)后,再次發(fā)起長(zhǎng)連接請(qǐng)求.
客戶(hù)端的代碼是像這樣的
一個(gè)簡(jiǎn)易的聊天室
通過(guò)長(zhǎng)連接, 我們可以開(kāi)發(fā)一個(gè)簡(jiǎn)易的web聊天室
下面, 我們通過(guò)redis開(kāi)發(fā)一個(gè)簡(jiǎn)易的web聊天室
1. 每一個(gè)客戶(hù)端發(fā)起長(zhǎng)連接時(shí), 在服務(wù)器端生成一個(gè)消息隊(duì)列, 對(duì)應(yīng)該用戶(hù). 然后監(jiān)聽(tīng)有無(wú)新數(shù)據(jù), 有則返回?cái)?shù)據(jù)到客戶(hù)端進(jìn)行處理, 并再起發(fā)起長(zhǎng)連接請(qǐng)求.
2. 每一個(gè)客戶(hù)端發(fā)起消息時(shí), 進(jìn)行消息隊(duì)列的廣播.
下面是代碼片段:
redisQueue = new RedisQueue(); $this->request = Request::createFromGlobals(); $this->response = new JsonResponse(); } public function on($event, Closure $closure) { if (is_callable($closure)) { $this->event[$event][] = $closure; } } public function fire($event) { if (isset($this->event[$event])) { foreach ($this->event[$event] as $callback) { call_user_func($callback, $this); } } } public function sendMessage($data) { switch ($data['type']) { case 'unicast': //單播 $this->unicast($data['target'], $data['data'], $data['resource']); break; case 'multicast': //組播 foreach ($data['target'] as $target) { $this->unicast($target, $data['data'], $data['resource']); } break; case 'broadcast': //廣播 foreach ($this->redisQueue->setQueueName('connections') as $target) { $this->unicast($target, $data['data'], $data['resource']); } break; } $this->fire('message'); } public function unicast($target, $message, $resource = 'system') { $redis_queue = new RedisQueue(); $redis_queue->setQueueName($target)->push($resource . ':' . $message); } public function getMessage($target) { return $this->redisQueue->setQueueName($target)->pop(); } public function hasMessage($target) { return count($this->redisQueue->setQueueName($target)); } public function run() { $data = $this->request->request; while (true) { if ($data->get('action') == 'getMessage') { if ($this->hasMessage($data->get('target'))) { $this->response->setData([ 'state' => 'ok', 'message' => '獲取成功', 'data' => $this->getMessage($data->get('target')) ]); $this->response->send(); break; } } elseif ($data->get('action') == 'connect') { $exist = false; foreach ($this->redisQueue->setQueueName('connections') as $connection) { if ($connection == $data->get('data')) { $exist = true; } } if (! $exist) { $this->redisQueue->setQueueName('connections')->push($data->get('data')); } $this->fire('connect'); break; } usleep(100000); } } }
長(zhǎng)連接避免了過(guò)于頻繁的輪詢(xún). 但服務(wù)器維持一個(gè)長(zhǎng)連接也有額外的資源消耗. 大并發(fā)時(shí)性能不理想. 在小型應(yīng)用里面可以考慮使用
更建議客戶(hù)端使用html5的websocket協(xié)議, 服務(wù)器端使用swoole.
感謝各位的閱讀!關(guān)于“PHP長(zhǎng)連接實(shí)現(xiàn)與使用方法的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!