zoukankan      html  css  js  c++  java
  • PHP 简单实现webSocket

    费话少说,用源代码说话

        1)客户端实现

      

    复制代码
     1 <html>
     2 <head>
     3     <meta charset="UTF-8">
     4     <title>Web sockets test</title>
     5     <script src="jquery-min.js" type="text/javascript"></script>
     6     <script type="text/javascript">
     7         var ws;
     8         function ToggleConnectionClicked() {          
     9                 try {
    10                     ws = new WebSocket("ws://127.0.0.1:2000");//连接服务器        
    11                     ws.onopen = function(event){alert("已经与服务器建立了连接
    当前连接状态:"+this.readyState);};
    12                     ws.onmessage = function(event){alert("接收到服务器发送的数据:
    "+event.data);};
    13                     ws.onclose = function(event){alert("已经与服务器断开连接
    当前连接状态:"+this.readyState);};
    14                     ws.onerror = function(event){alert("WebSocket异常!");};
    15                 } catch (ex) {
    16                     alert(ex.message);      
    17                 }
    18         };
    19  
    20         function SendData() {
    21             try{
    22                 var content = document.getElementById("content").value;
    23                 if(content){
    24                     ws.send(content);
    25                 }
    26                 
    27             }catch(ex){
    28                 alert(ex.message);
    29             }
    30         };
    31  
    32         function seestate(){
    33             alert(ws.readyState);
    34         }
    35        
    36     </script>
    37 </head>
    38 <body>
    39    <button id='ToggleConnection' type="button" onclick='ToggleConnectionClicked();'>连接服务器</button><br /><br />
    40    <textarea id="content" ></textarea>
    41     <button id='ToggleConnection' type="button" onclick='SendData();'>发送我的名字:beston</button><br /><br />
    42     <button id='ToggleConnection' type="button" onclick='seestate();'>查看状态</button><br /><br />
    43 
    44 </body>
    45 </html>
    复制代码

     2)服务器端实现

    复制代码
      1  class WS {
      2     var $master;  // 连接 server 的 client
      3     var $sockets = array(); // 不同状态的 socket 管理
      4     var $handshake = false; // 判断是否握手
      5 
      6     function __construct($address, $port){
      7         // 建立一个 socket 套接字
      8         $this->master = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)   
      9             or die("socket_create() failed");
     10         socket_set_option($this->master, SOL_SOCKET, SO_REUSEADDR, 1)  
     11             or die("socket_option() failed");
     12         socket_bind($this->master, $address, $port)                    
     13             or die("socket_bind() failed");
     14         socket_listen($this->master, 2)                               
     15             or die("socket_listen() failed");
     16 
     17         $this->sockets[] = $this->master;
     18 
     19         // debug
     20         echo("Master socket  : ".$this->master."
    ");
     21 
     22         while(true) {
     23             //自动选择来消息的 socket 如果是握手 自动选择主机
     24             $write = NULL;
     25             $except = NULL;
     26             socket_select($this->sockets, $write, $except, NULL);
     27 
     28             foreach ($this->sockets as $socket) {
     29                 //连接主机的 client 
     30                 if ($socket == $this->master){
     31                     $client = socket_accept($this->master);
     32                     if ($client < 0) {
     33                         // debug
     34                         echo "socket_accept() failed";
     35                         continue;
     36                     } else {
     37                         //connect($client);
     38                         array_push($this->sockets, $client);
     39                         echo "connect client
    ";
     40                     }
     41                 } else {
     42                     $bytes = @socket_recv($socket,$buffer,2048,0);
     43                     print_r($buffer);
     44                     if($bytes == 0) return;
     45                     if (!$this->handshake) {
     46                         // 如果没有握手,先握手回应
     47                         $this->doHandShake($socket, $buffer);
     48                         echo "shakeHands
    ";
     49                     } else {
     50                     
     51                         // 如果已经握手,直接接受数据,并处理
     52                         $buffer = $this->decode($buffer);
     53                         //process($socket, $buffer); 
     54                         echo "send file
    ";
     55                     }
     56                 }
     57             }
     58         }
     59     }
     60     
     61     function dohandshake($socket, $req)
     62     {
     63         // 获取加密key
     64         $acceptKey = $this->encry($req);
     65         $upgrade = "HTTP/1.1 101 Switching Protocols
    " .
     66                    "Upgrade: websocket
    " .
     67                    "Connection: Upgrade
    " .
     68                    "Sec-WebSocket-Accept: " . $acceptKey . "
    " .
     69                    "
    ";
     70 
     71         echo "dohandshake ".$upgrade.chr(0);           
     72         // 写入socket
     73         socket_write($socket,$upgrade.chr(0), strlen($upgrade.chr(0)));
     74         // 标记握手已经成功,下次接受数据采用数据帧格式
     75         $this->handshake = true;
     76     }
     77     
     78     
     79     function encry($req)
     80     {
     81         $key = $this->getKey($req);
     82         $mask = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
     83 
     84         return base64_encode(sha1($key . '258EAFA5-E914-47DA-95CA-C5AB0DC85B11', true));
     85     }
     86     
     87     function getKey($req) 
     88     {
     89         $key = null;
     90         if (preg_match("/Sec-WebSocket-Key: (.*)
    /", $req, $match)) { 
     91             $key = $match[1]; 
     92         }
     93         return $key;
     94     }
     95     
     96     // 解析数据帧
     97     function decode($buffer)  
     98     {
     99         $len = $masks = $data = $decoded = null;
    100         $len = ord($buffer[1]) & 127;
    101 
    102         if ($len === 126)  {
    103             $masks = substr($buffer, 4, 4);
    104             $data = substr($buffer, 8);
    105         } else if ($len === 127)  {
    106             $masks = substr($buffer, 10, 4);
    107             $data = substr($buffer, 14);
    108         } else  {
    109             $masks = substr($buffer, 2, 4);
    110             $data = substr($buffer, 6);
    111         }
    112         for ($index = 0; $index < strlen($data); $index++) {
    113             $decoded .= $data[$index] ^ $masks[$index % 4];
    114         }
    115         return $decoded;
    116     }
    117     
    118     // 返回帧信息处理
    119     function frame($s) 
    120     {
    121         $a = str_split($s, 125);
    122         if (count($a) == 1) {
    123             return "x81" . chr(strlen($a[0])) . $a[0];
    124         }
    125         $ns = "";
    126         foreach ($a as $o) {
    127             $ns .= "x81" . chr(strlen($o)) . $o;
    128         }
    129         return $ns;
    130     }
    131 
    132     // 返回数据
    133     function send($client, $msg)
    134     {
    135         $msg = $this->frame($msg);
    136         socket_write($client, $msg, strlen($msg));
    137     }
    138 }

    测试  $ws = new WS("127.0.0.1",2000);
    复制代码

     

  • 相关阅读:
    TCP
    JAVA面向对象
    windows本地搭建easy-mock环境
    创建axios拦截器
    记录axios高效率并发的方法
    JS:数组中push对象,覆盖问题
    使用el-tree-transfer的方式
    从后台拿到echarts的数据值,求出百分比
    项目遇到的小问题(关于vue-cli中js点击事件不起作用和iconfont图片下载页面css样式乱的解答)
    JavaScript 常用内置对象(字符串属性、Math对象、Array数组对象)
  • 原文地址:https://www.cnblogs.com/yzycoder/p/6721987.html
Copyright © 2011-2022 走看看