0

0

WebSocket使用笔记

php中文网

php中文网

发布时间:2016-07-29 09:05:51

|

1120人浏览过

|

来源于php中文网

原创

一、简介

对于网页中快速的发送接收多条消息,WebSocket非常适合来解决这种需求。此次使用WebSocket来搭建一个聊天网页应用。主要涉及的客户端的实现,服务端使用是一个php来实现的WebSocket服务端,在本篇中暂不做详细介绍。
主要的JavaScript代码有这些

  • new WebSocket 创建一个websocket对象
  • socket.onopen 建立连接后触发
  • socket.onerror 错误时触发
  • socket.onmessage 接受到数据时触发
  • socket.onclose 连接关闭时触发
  • socket.send() 发送数据

二、成果

最终实现的是向服务器发消息,服务器返回相同消息,效果如下图所示。在线展示版点击图片下发链接

WebSocket使用笔记

在线版本在这里

Cogram
Cogram

使用AI帮你做会议笔记,跟踪行动项目

下载

三、代码

JavaScript

var socket;
var submit = document.getElementById('submit');//首先输入聊天者名字,提交后可开始聊天var button = document.getElementById('send');//发送按钮
button.addEventListener('click', sendMsg);
submit.addEventListener('click', startChat);
functionstartChat(){
    document.getElementById('modal').style.visibility = "collapse";
    document.getElementById('modalBody').style.visibility = "collapse";
    connect();//连接服务器
}
//发送消息functionsendMsg(){var txetArea = document.getElementById('sendText');
    var content = txetArea.value;
    var client = document.getElementById('name').value;
    showMsg(content,0);
    txetArea.value = "";
    msg = {
        name: client,
        message: content
    }
    socket.send(JSON.stringify(msg)); //发送数据
}
//显示消息functionshowMsg(content,type){var mainDiv = document.getElementById('main');
    var div = document.createElement('div');
    mainDiv.appendChild(div);
    var lineDiv = mainDiv.lastChild;
    lineDiv.className = 'line';
    lineDiv.innerHTML = (function(){if (type == 0){
            return"<div class=\"selfContent\"></div>";
        }
        else {
            return"<div class=\"content\"></div>";
        }
    })();
    var contentDiv = lineDiv.lastChild;
    contentDiv.innerHTML = content;
    mainDiv.scrollTop = mainDiv.scrollHeight - mainDiv.clientHeight;
}
//建立连接functionconnect(){var http_request;
    if (window.XMLHttpRequest){ 
         http_request = new XMLHttpRequest();
         if (http_request.overrideMimeType){
            http_request.overrideMimeType('text/xml');
         }
    } elseif (window.ActiveXObject) { 
        try {
            http_request = new ActiveXObject("Msxml2.XMLHTTP");
        } catch (e){
            try {
                http_request = new ActiveXObject("Microsoft.XMLHTTP");
            } catch (e){}
        }
    }
    http_request.open('GET', "http://localhost/websocket/server.php", true);
    http_request.send();
    socket = new WebSocket("ws://localhost:9000/websocket/server.php");
    socket.onopen = open; //绑定成功打开后触发的函数
    socket.onmessage = recMessage; //绑定接受到数据后的处理函数
    socket.onerror = error; //绑定处理错误的函数
    socket.onclose = close; //绑定连接关闭后的处理函数
}
//连接成功functionopen(){var i = document.getElementsByTagName('i');
    i[0].innerHTML = '连接成功!';
    button.disabled = '';
}
//出错functionerror(){var i = document.getElementsByTagName('i');
    i[0].style.color = "#FF0000";
    i[0].innerHTML = '连接失败';
    button.disabled = 'disabled';
}
//接受到数据functionrecMessage(e){var data = JSON.parse(e.data);
    showMsg(data.message, 1);
}
//连接关闭functionclose(){
    button.disabled = 'disabled';
    if (confirm('连接已关闭,是否需要再次连接?')){
        connect();
    }
}

php服务端代码,已封装成类

<?phpnamespaceMyLab;

classWebSocket{private$host;
    private$port;
    function__construct($port, $host)
    {$this->host = $host;
        $this->port = $port;
    }
    //启动服务functionStartServer(){$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);//创建socket
        socket_set_option($socket, SOL_SOCKET, SO_REUSEADDR, 1);
        socket_bind($socket, 0, $this->port);
        socket_listen($socket);//开启监听$sockets = array($socket);
        $write = NULL;
        $except = NULL;
        while (true)
        {
            $clients = $sockets;
            $num = socket_select($clients, $write, $except, 0);
            if ($num === false){
                echo"socket_select failed!";
                break;
            }
            if (in_array($socket, $clients))
            {
                $socket_new = socket_accept($socket); //接受连接$sockets[] = $socket_new; //保存连接$header =  socket_read($socket_new, 1024);
                $status = $this->handshaking($header, $socket_new, $this->host, $this->port);
                $response = $this->code(json_encode(array('message'=>'欢迎来到Chat with yourself')));
                $status = $this->sendMessage($response, $socket_new);//首次连接上之后回应$found_socket = array_search($socket, $clients);
                unset($clients[$found_socket]);
            }
            foreach ($clientsas$client) //遍历连接,处理接受到的消息
            {
                while (socket_recv($client, $buf, 1024, 0) >= 1)
                {
                    $received_text = $this->decode($buf);
                    $decode_text = json_decode($received_text);
                    $user_name = $decode_text->name;
                    $user_message = $decode_text->message;
                    $response_text = $this->code(json_encode(array('type' => 'usermsg', 'name' => $user_name, 'message' => $user_message)));
                    $this->sendMessage($response_text, $client);
                    break2;
                }
            }
        }
    }
    //握手验证functionhandshaking($receved_header,$client_conn, $host, $port)
    {$headers = array();
        $lines = preg_split("/\r\n/", $receved_header);
        foreach($linesas$line)
        {
            $line = chop($line);
            if(preg_match('/\A(\S+): (.*)\z/', $line, $matches))
            {
                $headers[$matches[1]] = $matches[2];
            }
        }

        $secKey = $headers['Sec-WebSocket-Key'];
        $secAccept = base64_encode(pack('H*', sha1($secKey . '258EAFA5-E914-47DA-95CA-C5AB0DC85B11')));
        $upgrade  = "HTTP/1.1 101 Web Socket Protocol Handshake\r\n" .
        "Upgrade: websocket\r\n" .
        "Connection: Upgrade\r\n" .
        "WebSocket-Origin: $host\r\n" .
        "WebSocket-Location: ws://$host:$port/demo/shout.php\r\n".
        "Sec-WebSocket-Accept:$secAccept\r\n\r\n";
        socket_write($client_conn,$upgrade,strlen($upgrade));
        return$upgrade;
    }
    //解码functiondecode($str)
    {$length = ord($str[1]) & 127;
        if ($length == 126)
        {
            $masks = substr($str, 4, 4);
            $data = substr($str, 8);
        }
        elseif ($length == 127) {
            $masks = substr($str, 10, 4);
            $data = substr($str, 14);
        }
        else {
            $masks = substr($str, 2, 4);
            $data = substr($str, 6);
        }
        $str = '';
        for ($i = 0; $i < strlen($data); ++$i)
        {
            $str .= $data[$i] ^ $masks[$i % 4];
        }
        return$str;
    }
    //发送消息functionsendMessage($msg, $cilent)
    {try{

            return socket_write($cilent, $msg, strlen($msg));
        }
        catch (\Exception$e){
            return$e;
        }
    }
    //编码functioncode($str)
    {$b1 = 0x80 | (0x1 & 0x0f);
        $length = strlen($str);
        if ($length <= 125)
            $header = pack('CC', $b1, $length);
        elseif ($length >125 && $length < 65536)
            $header = pack('CCn', $b1, 126, $length);
        elseif ($length >= 65536)
            $header = pack('CCNN', $b1, 127, $length);
        else {
            returnfalse;
        }
        return$header.$str;
    }

}

').addClass('pre-numbering').hide(); $(this).addClass('has-numbering').parent().append($numbering); for (i = 1; i ').text(i)); }; $numbering.fadeIn(1700); }); });

以上就介绍了WebSocket使用笔记,包括了方面的内容,希望对PHP教程有兴趣的朋友有所帮助。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

705

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

233

2026.02.13

Flutter跨平台开发与状态管理实战
Flutter跨平台开发与状态管理实战

本专题围绕Flutter框架展开,系统讲解跨平台UI构建原理与状态管理方案。内容涵盖Widget生命周期、路由管理、Provider与Bloc状态管理模式、网络请求封装及性能优化技巧。通过实战项目演示,帮助开发者构建流畅、可维护的跨平台移动应用。

117

2026.02.13

TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

22

2026.02.13

Redis高可用架构与分布式缓存实战
Redis高可用架构与分布式缓存实战

本专题围绕 Redis 在高并发系统中的应用展开,系统讲解主从复制、哨兵机制、Cluster 集群模式及数据分片原理。内容涵盖缓存穿透与雪崩解决方案、分布式锁实现、热点数据优化及持久化策略。通过真实业务场景演示,帮助开发者构建高可用、可扩展的分布式缓存系统。

61

2026.02.13

c语言 数据类型
c语言 数据类型

本专题整合了c语言数据类型相关内容,阅读专题下面的文章了解更多详细内容。

30

2026.02.12

雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法
雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法

本专题系统整理雨课堂网页版官方入口及在线登录方式,涵盖账号登录流程、官方直连入口及平台访问方法说明,帮助师生用户快速进入雨课堂在线教学平台,实现便捷、高效的课程学习与教学管理体验。

15

2026.02.12

豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法
豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法

本专题汇总豆包AI官方网页版入口及在线使用方式,涵盖智能写作工具、图片生成体验入口和官网登录方法,帮助用户快速直达豆包AI平台,高效完成文本创作与AI生图任务,实现便捷智能创作体验。

669

2026.02.12

PostgreSQL性能优化与索引调优实战
PostgreSQL性能优化与索引调优实战

本专题面向后端开发与数据库工程师,深入讲解 PostgreSQL 查询优化原理与索引机制。内容包括执行计划分析、常见索引类型对比、慢查询优化策略、事务隔离级别以及高并发场景下的性能调优技巧。通过实战案例解析,帮助开发者提升数据库响应速度与系统稳定性。

58

2026.02.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.4万人学习

golang socket 编程
golang socket 编程

共2课时 | 0.1万人学习

swoole入门物联网开发与实战
swoole入门物联网开发与实战

共15课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号