设为首页 收藏本站
查看: 967|回复: 0

[经验分享] PHP+WebSocket搭建简易聊天室实践

[复制链接]
累计签到:16 天
连续签到:1 天
发表于 2016-10-31 14:48:49 | 显示全部楼层 |阅读模式
1、前言  公司游戏里面有个简单的聊天室,了解了之后才知道是node+websocket做的,想想php也来做个简单的聊天室。于是搜集各种资料看文档、找实例自己也写了个简单的聊天室。
  http连接分为短连接和长连接。短连接一般可以用ajax实现,长连接就是websocket。短连接实现起来比较简单,但是太过于消耗资源。websocket高效不过兼容存在点问题。websocket是html5的资源
  本文主要介绍websocket简易聊天室的实现步骤具体部分知识点的深入会给出链接或者麻烦读者自己搜集资料。
2、前端
  前端实现websocket很简单直接
  //连接websocket
      var ws = new WebSocket("ws://127.0.0.1:8000");
  //成功连接websoc的时候
  ws.onopen = function(){}
  //成功获取服务端输出的消息
  ws.onmessage = function(e){}
     //连接错误的时候
  ws.onerror = function(){}

    //向服务端发送数据
  ws.send();
3、后台
  websocket的难点主要在后台
  3.1websocket连接过程
  websocket 通信图解 这是一个简易的客户端和服务端的通信图解,php主要就做的就是接受加密key  并返回 其中完成套接字的创建和握手操作
   161031080426642.jpg
    下图是一张详细的服务端处理websocket的流程图
   161031080426643.jpg

3.2 代码实践
  服务端做的流程大致是:
    ①、挂起一个socket套接字进程等待连接
    ②、有socket连接之后遍历套接字数组
    ③、没有握手的进行握手操作,如果已经握手则接收数据解析并写入缓冲区进行输出
  下面是示例代码(我写的是一个类所以代码是根据函数分段的),文底给出github地址以及自己遇到的一些坑
     1、首先是创建套接字
//建立套接字        public function createSocket($address,$port)        {            //创建一个套接字            $socket= socket_create(AF_INET, SOCK_STREAM, SOL_TCP);            //设置套接字选项            socket_set_option($socket, SOL_SOCKET, SO_REUSEADDR, 1);            //绑定IP地址和端口            socket_bind($socket,$address,$port);            //监听套接字            socket_listen($socket);            return $socket;        }
  2、将套接字放入数组
public function  __construct($address,$port)        {            //建立套接字            $this->soc=$this->createSocket($address,$port);            $this->socs=array($this->soc);        }
3、挂起进程遍历套接字数组,主要操作都是在这里面完成的
public function run(){            //挂起进程            while(true){                $arr=$this->socs;                $write=$except=NULL;                //接收套接字数字 监听他们的状态                socket_select($arr,$write,$except, NULL);                //遍历套接字数组                foreach($arr as $k=>$v){                    //如果是新建立的套接字返回一个有效的 套接字资源                    if($this->soc == $v){                        $client=socket_accept($this->soc);                        if($client <0){                            echo "socket_accept() failed";                        }else{                            // array_push($this->socs,$client);                            // unset($this[]);                            //将有效的套接字资源放到套接字数组                            $this->socs[]=$client;                        }                    }else{                        //从已连接的socket接收数据  返回的是从socket中接收的字节数                        $byte=socket_recv($v, $buff,20480, 0);                        //如果接收的字节是0                        if($byte<7)                            continue;                        //判断有没有握手没有握手则进行握手,如果握手了 则进行处理                        if(!$this->hand[(int)$client]){                            //进行握手操作                            $this->hands($client,$buff,$v);                        }else{                            //处理数据操作                            $mess=$this->decodeData($buff);                               //发送数据                            $this->send($mess,$v);                        }                    }                }            }        }
4、进行握手 流程是接收websocket内容从Sec-WebSocket-Key:中获取key并通过加密算法写入缓冲区客户端会进行验证(自动验证不需要我们处理)
public function hands($client,$buff,$v)        {            //提取websocket传的key并进行加密  (这是固定的握手机制获取Sec-WebSocket-Key:里面的key)            $buf  = substr($buff,strpos($buff,'Sec-WebSocket-Key:')+18);            //去除换行空格字符            $key  = trim(substr($buf,0,strpos($buf,"\r\n")));             //固定的加密算法            $new_key = base64_encode(sha1($key."258EAFA5-E914-47DA-95CA-C5AB0DC85B11",true));            $new_message = "HTTP/1.1 101 Switching Protocols\r\n";            $new_message .= "Upgrade: websocket\r\n";            $new_message .= "Sec-WebSocket-Version: 13\r\n";            $new_message .= "Connection: Upgrade\r\n";            $new_message .= "Sec-WebSocket-Accept: " . $new_key . "\r\n\r\n";            //将套接字写入缓冲区            socket_write($v,$new_message,strlen($new_message));            // socket_write(socket,$upgrade.chr(0), strlen($upgrade.chr(0)));            //标记此套接字握手成功            $this->hand[(int)$client]=true;        }
5、解析客户端的数据(我这里没有进行加密,如果有需要也可以自己加密 )
//解析数据        public  function  decodeData($buff)        {            //$buff  解析数据帧            $mask = array();              $data = '';              $msg = unpack('H*',$buff);  //用unpack函数从二进制将数据解码            $head = substr($msg[1],0,2);              if (hexdec($head{1}) === 8) {                  $data = false;              }else if (hexdec($head{1}) === 1){                  $mask[] = hexdec(substr($msg[1],4,2));                  $mask[] = hexdec(substr($msg[1],6,2));                  $mask[] = hexdec(substr($msg[1],8,2));                  $mask[] = hexdec(substr($msg[1],10,2));                     //遇到的问题  刚连接的时候就发送数据  显示 state connecting                $s = 12;                  $e = strlen($msg[1])-2;                  $n = 0;                  for ($i=$s; $i<= $e; $i+= 2) {                      $data .= chr($mask[$n%4]^hexdec(substr($msg[1],$i,2)));                      $n++;                  }                //发送数据到客户端                   //如果长度大于125 将数据分块                   $block=str_split($data,125);                   $mess=array(                       'mess'=>$block[0],                       );                return $mess;                               }[url=] 161031080426641.jpg [/url]


6、将套接字写入缓冲区
[url=] 161031080426641.jpg [/url]
//发送数据        public function send($mess,$v)        {            //遍历套接字数组 成功握手的  进行数据群发            foreach ($this->socs as $keys => $values) {                //用系统分配的套接字资源id作为用户昵称                   $mess['name']="Tourist's socket:{$v}";                   $str=json_encode($mess);                   $writes ="\x81".chr(strlen($str)).$str;                   // ob_flush();                   // flush();                   // sleep(3);                   if($this->hand[(int)$values])                       socket_write($values,$writes,strlen($writes));               }        }
7、运行方法
github地址git@github.com:rsaLive/websocket.git
①最好在控制台运行server.php
转到server.php脚本目录(可以先php -v 看下有没有配置php如果没有Linux配置下bash windows 配置下path)
php -f server.php
161031080426644.jpg
如果有错误会提示
161031080426645.jpg
②通过服务器访问html文件
161031080426646.jpg
161031080426647.jpg

8、踩过的坑,打开调试工作方便查看错误
server.php 挂起的进程中可以打印输出的,如果出现问题可以在代码中加入打印来调试
可以在各个判断里面做标记在控制台查看代码运行在哪个区间
不过每次修改完代码之后需要重新运行脚本 php server.php

如果出现这种错误可能是
161031080426648.jpg
  1、在与服务器初始套接字的时候发送数据 (在第一次与服务器验证握手的时候不能发送内容)
  2、如果已经验证过了但是客户端没有发送或者发送的消息为空也会出现这样的情况
    所以要检验已连接的套接字的数据
161031080426649.jpg
③可能浏览器不支持或者服务端没有开启socket开始之前最好验证下
if (window.WebSocket){    console.log("This browser supports WebSocket!");} else {    console.log("This browser does not support WebSocket.");}
如有不正欢迎指出


运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-293772-1-1.html 上篇帖子: 编译安装php5.4.45 下篇帖子: php批量删除cookie的简单实现方法
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表