前言
????前面有寫一個異步簡單的聊天室,然后想著,就把協程的也弄了吧所以就有了這個文章,其實所有的功能都大差不差,就僅僅幾個地方不一樣而已,也都是簡單的地方。
博文地址:利用 websocket 與 swoole 打造一個小型聊天室 (異步)
本次也沒增加功能,就是增加了一個心跳,從前端定時發送一個 ping ,服務端不作反應,僅此而已。
前端頁面代碼:
nbsp;html> ???? <meta>???? <title>打工人聊天室</title>??? <!--需要引入jq 文件--><style> .content { height: 400px; max-width: 400px; overflow: auto; border-radius: 5px; border: 1px solid #f0f0f0; }</style>???????????? <div>???????????????? <p>聊天區域</p>???????????? </div>???????????? 你好打工人:<samp>昵稱</samp>?<br>???????????? 本次連接FD:?<samp></samp>?<br>???????????? <input>???????????? <input>???????????? <button>發送</button>
JS 代碼:
????在服務器信息回執時,會有第一次連接回執,還是服務端發送消息回執的狀態區別,通過 msgType 來分辨,如果是第一次連接的回執消息,則把 FD 做一個頁面留存,并不顯示在聊天消息區,如果收到的是消息回執,就直接顯示到聊天消息區。
????還有就是,前后端相互通信發送的東西,都是字符串性質最優,我前端處理的方法是先組合成一個對象,然后轉 JSON 串。
<script> //滾動條最底部 function scrolltest() { var div = document.getElementById("content"); div.scrollTop = div.scrollHeight; } var wsServer = 'ws://127.0.0.1:9502/websocket'; var websocket = new WebSocket(wsServer); var nickname = Math.random().toString(36).substr(2); thisFd = ''; $('#nickname').html(nickname); //點擊發送 function send() { var msg = $('#msg').val(); var data = { 'nickname': nickname, 'fd': thisFd, 'data': msg } //生成json 方便后臺接收以及使用 var data = JSON.stringify(data); websocket.send(data); //然后清空 $('#msg').val(''); } //鏈接成功 websocket.onopen = function (evt) { var data = { 'msgType': 'open' } var data = JSON.stringify(data); $("#content >p:last-child").after('<p> 服務器已連接,開始聊天吧 '); websocket.send(data); }; //鏈接斷開 websocket.onclose = function (evt) { $("#content >p:last-child").after('<p> 服務器已斷開,請重新連接 '); }; //收到服務器消息 websocket.onmessage = function (evt) { //握手成功后,會接受到服務端返回的fd ,msgType = 1 //字符串格式化成json var data = eval('(' + evt.data + ')'); // console.log(evt.data); switch (data.msgType) { case 1: thisFd = data.fd; $('#fd-samp').html(thisFd); $('#fd').val(thisFd); break; case 2: if (data.nickname == nickname) { data.nickname = '我'; } $("#content >p:last-child").after('<p>' + data.nickname + ' 在 ' + data.time + ' 說:<br>' + data.data + ''); //接收到消息自動觸底 scrolltest(); break; } }; //服務器異常 websocket.onerror = function (evt, e) { $("#content >p:last-child").after('<p> 服務器異常 '); }; //心跳,本次新增 function heartbeat() { var data = { 'msgType': 'ping', } //生成json 方便后臺接收以及使用 var data = JSON.stringify(data); websocket.send(data); } //30 秒一次 setInterval(heartbeat, 30000);</script>
服務端代碼
協程,都需要在 Corun(function () {}) 里。
<?php //定義獲取當前的id函數 function getObjectId(swooleHttpResponse $response) { if (PHP_VERSION_ID < 70200) { $id = spl_object_hash($response); } else { $id = spl_object_id($response); } return $id; } Corun(function () { $server = new CoHttpServer('127.0.0.1', 9502, false); $server->set([???????????? 'heartbeat_idle_time'??????=>?600,? //?表示一個連接如果600秒內未向服務器發送任何數據,此連接將被強制關閉???????????? 'heartbeat_check_interval'?=>?60,?? //?表示每60秒遍歷一次???????? ]);???????? $server->handle('/websocket',?function?($request,?$ws)?{???????????? $ws->upgrade();???????????? global?$wsObjects;???????????? $objectId?=?getObjectId($ws);???????????? $wsObjects[$objectId]?=?$ws;???????????? while?(true)?{???????????????? $frame?=?$ws->recv();???????????????? if?($frame?===?'')?{???????????????????? unset($wsObjects[$objectId]);???????????????????? $ws->close();???????????????????? break;???????????????? }?else?if?($frame?===?false)?{???????????????????? echo?'error?:?'?.?swoole_last_error()?.?"n";???????????????????? break;????????????????}? else?{???????????????????? if?($frame->data?==?'close'?||?get_class($frame)?===?SwooleWebSocketCloseFrame::class)? {???????????????????????? unset($wsObjects[$objectId]);???????????????????????? $ws->close();???????????????????????? return;???????????????????? }???????????????????? //格式化接收到json???????????????????? $data?=?json_decode($frame->data);???????????????????? switch?($data->msgType){???????????????????????? case?'open':???????????????????????????? //鏈接第一次???????????????????????????? $data?=?json_encode([???????????????????????????????? 'fd'?=>?$objectId,???????????????????????????????? 'msgType'?=>?1?? //代表第一次連接,前端處理fd???????????????????????????? ]);???????????????????????????? $ws->push($data);???????????????????????????? break;???????????????????????? case?'ping':???????????????????????????? //接收到心跳?不作回復 //???????????????????????????? echo??$data->msgType;???????????????????????????? break;???????????????????????? default?:???????????????????????????? //?原基礎上不動,增加一些自定義???????????????????????????? $data->msgType?=?2;? //代表服務器端回復???????????????????????????? $data->time?=?date('Y-m-d?H-i-s');???????????????????????????? $data?=?json_encode($data);???????????????????????????? foreach?($wsObjects?as?$obj)?{???????????????????????????????? $obj->push($data);???????????????????????????? }???????????????????? }???????????????? }????????????}???????? });???????? $server->start();???? });
代碼齊全之后,接下來就只需要在控制臺執行以下 PHP 文件就行。
然后前臺直接訪問你的網站地址,我的是本地 127.0.0.1
多開幾個窗口模擬多個用戶,然后發送消息測試即可:
你好,打工人。
代碼很簡單,難度不大,但是可以很簡潔的反應出 webScoket 和 Swoole 的一種強大。
? 版權聲明
文章版權歸作者所有,未經允許請勿轉載。
THE END