前言

详解 HTTP系列之一讲到HTTP/2.0 突破了传统的“请求-问答模式”这一局限,实现了服务器主动向客户端传送数据。而本章将通过一种在单个TCP连接上进行全双工通信的协议--websocket协议,实现一个简单的聊天室(本文转载自xing.org1^大佬的博客,详情请点击此处跳转




核心要点

聊天室的核心要点是由服务器向每个正在连接的用户发送消息,也就是上面讲到的改变。如果不是这样,只能是客户端以一个很短的时间间隔向服务端请求数据。

要做到广播,就需要server.connections,这个数组记录了所有连接到websocket服务器的用户(也就是进入聊天室的人),通过遍历这个数组,然后给数组中每个连接进来的用户对象发送消息即可。

实现这一功能的代码如下:

const ws = require('nodejs-websocket');
const server = ws.createServer((connect)=>{
/*
......
*/ // 对连接到服务端的每个对象发送数据
function broadcast(jsonStr){
server.connections.forEach((element)=>{ //切记,send参数必须是字符串类型,所以这里将对象字符串化
element.send(JSON.stringify(jsonStr));
});
}
})

 

 

聊天室实现代码

环境:

  • nodejs
  • npm install nodejs-websocket

 

客户端 test.html:

<!DOCTYPE html>
<html lang="en"> <head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>小石头的群聊</title>
</head> <body>
<div class="container">
<input type="text" placeholder="请输入要发送的内容" id="ipt">
<button id="btn">发送</button>
<button id="closeBtn">退出群聊</button>
<p>
群聊内容如下:
</p>
<div id="rst" style="width: 800px;height: 500px;background-color: greenyellow;"></div>
</div>
<script>
// 聊天室功能
const ws = new WebSocket('ws://localhost:8081'),
ipt = document.getElementById('ipt'),
btn = document.getElementById('btn'),
closeBtn = document.getElementById('closeBtn'),
content = document.getElementById('rst'); // 初次进入聊天室,给个提醒
ws.addEventListener('open', (e) => {
console.log('加入聊天室成功', e);
content.appendChild(creatEle('您已成功加入小石头的群聊~')); //
// 聊天区滚动到底
goBottom();
}); // 接收ws服务器发送的消息,并展示到div#rst当中
ws.addEventListener('message', (e) => {
console.log(e.data);
content.appendChild(creatEle(e.data));
goBottom();
}); // 一个带滚动条的DIV元素,怎么让它的滚动条位置默认保持在最底部?
function goBottom() {
content.scrollTop = content.scrollHeight;
} // btn被点击时发送请求
btn.onclick = function () {
btnClickEvent();
}
ipt.addEventListener('keydown', (e) => {
if (e.keyCode === 13) {
btnClickEvent();
}
}); // close-btn被点击时退出群聊
closeBtn.onclick = function () {
ws.close();
} // 创建一个p标签,存储对应内容,以追加到内容展示区域
function creatEle(str) {
console.log(str.indexOf('{'));
const TYPE_LEAVE = 0; //leave,离开
const TYPE_ENTRY = 1; //entry,进入
const TYPE_SPEAK = 2; //speak,发言
const eleP = document.createElement('p'); if (str.indexOf('{') == 0) {
let parseStr = JSON.parse(str);
eleP.innerHTML = `<span class="timer">${parseStr.time}</span><br/><span class="msg">${parseStr.msg}</span>`;
switch (parseStr.type) {
case TYPE_LEAVE:
eleP.className = 'leave';
break;
case TYPE_ENTRY:
eleP.className = 'entry';
break;
case TYPE_SPEAK:
eleP.className = 'speek';
break;
default:
eleP.className = 'default';
break;
}
} else {
eleP.innerText = str;
}
return eleP;
}
// 添加回车发送消息事件
function btnClickEvent() {
if (ipt.value.length <= 0) {
alert('不能发送空消息');
return;
}
ws.send(ipt.value);
ipt.value = '';
}
</script>
</body>
</html>

 

服务端 server.js:

/*
* @Author: @Guojufeng
* @Date: 2019-06-02 19:42:06
* @Last Modified by: @Guojufeng
* @Last Modified time: 2019-06-02 21:39:56
* 优化 - 加入消息类型和当前时间的响应
*/
const ws = require('nodejs-websocket');
const POST = 8081; let count = 0;//记录加入人数 const TYPE_LEAVE = 0;//leave,离开
const TYPE_ENTRY = 1;//entry,进入
const TYPE_SPEAK = 2;//speak,发言 const server = ws.createServer((connect)=>{
count++;// 有人加入,计数加一
connect.userName = `用户${count}`;//connet是一个对象,新增一个属性用以标记该用户的名字(标识) // 1、通知所有人,connect用户加入群聊
broadcast({
type: TYPE_ENTRY,
msg: `${connect.userName}加入群聊。`
}); // 2、通知所有人,connect用户发言
connect.on('text',(rst)=>{
broadcast({
type: TYPE_SPEAK,
msg: `${connect.userName}: ${rst}`
});
}); // 3、通知所有人,connect用户退出群聊
connect.on('close',(code,reason)=>{//个人认为利用code和reason这里,还可以模拟微信群聊中,用户被群主踢出群的情况
broadcast({
type: TYPE_LEAVE,
msg: `${connect.userName}退出了群聊。`
});
count--;// 有人退出,计数减一
}); // error事件
connect.on('error',()=>{
console.log('发生异常');
})
}); // 广播功能代码
function broadcast(cont){
// 利用connections是存放所有加入群聊用户的数组,来给所有人广播内容
let sendCont = {
type: cont.type,
msg: cont.msg,
time: new Date().toLocaleTimeString()
};
server.connections.forEach((element)=>{
element.send(JSON.stringify(sendCont));//切记,send参数必须是字符串类型,所以这里将对象字符串化
});
} server.listen(POST,()=>{
console.log(`${POST}服务器启动成功。`)
});

启动服务:

  1. nodejs执行server.js文件,并成功监听到对应端口
  2. 浏览器打开index.html(可开启多个页面模拟多个用户),进行测试




结语


时间:2020/08/16 11:10

坐标:广东深圳


小案例-WebSocket实现简易聊天室的更多相关文章

  1. 基于Node.js + WebSocket 的简易聊天室

    代码地址如下:http://www.demodashi.com/demo/13282.html Node.js聊天室运行说明 Node.js的本质就是运行在服务端的JavaScript.Node.js ...

  2. node.js+websocket实现简易聊天室

    (文章是从我的个人主页上粘贴过来的,大家也可以访问我的主页 www.iwangzheng.com) websocket提供了一种全双工客户端服务器的异步通信方法,这种通信方法使用ws或者wss协议,可 ...

  3. php+websocket搭建简易聊天室实践

    1.前言 公司游戏里面有个简单的聊天室,了解了之后才知道是node+websocket做的,想想php也来做个简单的聊天室.于是搜集各种资料看文档.找实例自己也写了个简单的聊天室. http连接分为短 ...

  4. node+websocket创建简易聊天室

    关于websocket的介绍太多,在这就不一一介绍了,本文主要实现通过websocket创建一个简易聊天室,就是90年代那种聊天室 服务端 1.安装ws模块,uuid模块,ws是websocket模块 ...

  5. 使用Html5下WebSocket搭建简易聊天室

    一.Html5WebSocket介绍 WebSocket protocol 是HTML5一种新的协议(protocol).它是实现了浏览器与服务器全双工通信(full-duplex). 现在,很多网站 ...

  6. WebSocket实现简易聊天室

    前台页面: <html> <head> <meta http-equiv="Content-Type" content="text/html ...

  7. 基于WebSocket的简易聊天室

    用的是Flash + WebSocket 哦~ Flask 之 WebSocket 一.项目结构: 二.导入模块 pip3 install gevent-websocket 三.先来看一个一对一聊天的 ...

  8. Java WebSocket实现简易聊天室

    一.Socket简介 Socket又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请求.Socket的英文原义是“孔”或“插座”,作为UNI ...

  9. Servlet WebSocket的简易聊天室

    添加依赖 <!-- websocket --> <dependency> <groupId>javax.websocket</groupId> < ...

随机推荐

  1. java.lang.ClassNotFoundException: org.apache.tomcat.util.security.Escape

    tomcat-embed-jasper 依赖中不要有版本号 技术交流群: 816227112

  2. 性能分析(3)- 短时进程导致用户 CPU 使用率过高案例

    性能分析小案例系列,可以通过下面链接查看哦 https://www.cnblogs.com/poloyy/category/1814570.html 系统架构背景 VM1:用作 Web 服务器,来模拟 ...

  3. SpringBoot+Dynamic多数据源动态切换

    最近做了个小模块,需求就是项目同时读取三个数据库,操作数据.并不是分库分表,只用定时跑,不需要对外提供接口. 技术选型:SpringBoot + Mybatis Plus(Mybatis) + Dyn ...

  4. static关键字设计原理

    语法只是表象,原理才是关键!!! 灵魂static关键字 Java规定:方法只能由对象来调用. 换句话来说,在面向对象的思维下,方法与对象存在一种强耦合. 方法在没有对象的情况下无法调用,于是上帝派来 ...

  5. day18.os模块 对系统进行操作

    一.os操作 1.system()在python中执行系统命令 # os.system("ifconfig") # os.system("touch 1.txt" ...

  6. c++ explict

    explicit 用于一个参数的构造函数:防止隐式转换. 什么意思呢? myClass(int x); 这是个构造函数 我们可以使用 myClass a(4);  或 myClass a=4;来调用它 ...

  7. 阿里ECS云服务器部署文件

    今天,接触了阿里ECS云服务器,免费领取链接https://dwz.cn/WOFZpZz1 获取之后,要添加一下端口,刚开始需要80  8080  3306的端口,其他的根据需要自行添加 点击快速创建 ...

  8. 蓝奏云数值验证码识别,python调用虹鱼图灵识别插件,超高正确率

    识别验证码一直是本人想要做的事情,一直在接触按键精灵,了解到有一个虹鱼图灵识别插件专门做验证码和图像识别,原理就是图片处理和制作字库识别,制作字库我一直觉得很麻烦,工程量太大.不管怎样,它能用能达到我 ...

  9. pageHelper使用时的注意点

    1 在pom.xml中导入相关的依赖(注意版本问题,报错十有八九是因为版本问题) <dependency> <groupId>com.github.pagehelper< ...

  10. ConHost.exe机制