使用Html5下WebSocket搭建简易聊天室
一、Html5WebSocket介绍
WebSocket protocol 是HTML5一种新的协议(protocol)。它是实现了浏览器与服务器全双工通信(full-duplex)。
现在,很多网站为了实现即时通讯(real-time),所用的技术都是轮询(polling)。轮询是在特定的的时间间隔(time interval)(如每1秒),由浏览器对服务器发出HTTP request,然后由服务器返回最新的数据给客服端的浏览器。这种传统的HTTP request d的模式带来很明显的缺点 – 浏览器需要不断的向服务器发出请求(request),然而HTTP request 的header是非常长的,里面包含的数据可能只是一个很小的值,这样会占用很多的带宽。
而最比较新的技术去做轮询的效果是Comet – 用了AJAX。但这种技术虽然可达到全双工通信,但依然需要发出请求(reuqest)。
在 WebSocket API,浏览器和服务器只需要要做一个握手的动作(实际上是tcp),然后,浏览器和服务器之间就形成了一条快速通道(这里走的是新的协议)。两者之间就直接可以数据互相传送。
二、IM系统的几种通信方式
1.点对点通信(对等通信方式):客户端A想要与客户端B进行通信,首页会与IM服务器进行一次握手,然后从IM服务器拿到客户端B的地址。然后直接向客户端B发送消息,然后客户端B获取到A客户端的地址,也直接向客户端A回复消息,这样就不通过IM服务器来中转,这样双方的即时文字消息就不通过 IM服务器中转,而是通过网络进行点对点的直接通讯,这称为对等通讯方式(Peer To Peer) 。PS:这种方式需要做内网穿透或代理,不然无法获取到对方的地址等信息。
2.代理通信:当客户端A与客户端B之间存在防火墙,网速很慢等原因,IM服务器可以提供消息中专的服务,客户端A先把消息发送到IM服务器,然后再通过IM服务器把消息转发给客户端B,这样无需得到客户端的地址信息就能实现消息送达,这种方式叫做代理通信。
3.离线代理通信:当客户端A想要与客户端B通信的时候,发现客户端B不在线,这样IM服务器会把消息存起来,等到下一次客户端B上线的时候,由客户端B主动获取到离线消息(这样做好像可以降低服务器的压力)。
三、利用Html5的WebSocket实现简单的聊天室
1.服务端代码如下,注释那些都挺全的,就不一一多说:
private async Task WebSocketContext(AspNetWebSocketContext context)
{
try
{
WebSocket socket = context.WebSocket; //获取连接信息
string user_name = TDCMS.Common.TD_Request.GetQueryStringValue("user_name", ""); //第一次open时,添加到连接池中
if (_userPool.Find(c => c.User_name== user_name) == null)
{
_userPool.Add(new UserPool() { User_name = user_name , Socket = socket });
}
else
{
UserPool p = _userPool.Find(c => c.User_name == user_name);
if (socket != p.Socket)//当前对象不一致,更新
{
p.Socket = socket;
}
} UserPool sourcePool= _userPool.Find(c => c.User_name == user_name);//获取到发送者连接池 #region 对所有连接池中广播 我上线了
foreach (var item in _userPool)
{
MessageModel model = new MessageModel()
{
Aim = item.User_name,
Contents = user_name + "上线了",
Source = sourcePool.User_name,
Status =
};
await item.Socket.SendAsync(new ArraySegment<byte>(Encoding.UTF8.GetBytes(JsonHelper.ObjectToJson(model))), WebSocketMessageType.Text, true, CancellationToken.None);
}
#endregion bool isNext = true;
while (isNext)
{
if (socket.State == WebSocketState.Open)
{
ArraySegment<byte> buffer = new ArraySegment<byte>(new byte[]);
WebSocketReceiveResult result = await socket.ReceiveAsync(buffer, CancellationToken.None); #region 关闭Socket处理,删除连接池
if (socket.State != WebSocketState.Open)//连接关闭
{
if (_userPool.Find(c => c.User_name == user_name) != null)
_userPool.Remove(_userPool.Find(c => c.User_name == user_name));//删除连接池
//广播当前在线的用户 我下线了
foreach (var item in _userPool)
{
MessageModel offline = new MessageModel()
{
Aim = item.User_name,
Contents = user_name + "下线了",
Source = sourcePool.User_name,
Status =
};
await item.Socket.SendAsync(new ArraySegment<byte>(Encoding.UTF8.GetBytes(JsonHelper.ObjectToJson(offline))), WebSocketMessageType.Text, true, CancellationToken.None);
}
break;
} #endregion #region 如果连接没有关闭,处理发送过来的消息 MessageModel model=new MessageModel();
int messageCount = result.Count;
string messageStr= Encoding.UTF8.GetString(buffer.Array, , messageCount);
model = JsonHelper.JsonToObject<MessageModel>(messageStr);//这个是解析好的 消息 //发送消息到每个客户端
foreach (var item in _userPool)
{
await item.Socket.SendAsync(new ArraySegment<byte>(Encoding.UTF8.GetBytes(JsonHelper.ObjectToJson(model))), WebSocketMessageType.Text, true, CancellationToken.None);
} #endregion
}
} }
catch(Exception ex)
{
throw ex;
}
}
2.客户端JS代码如下:
<script>
var im;//WebSocket对象
function initIm() {
var user=$('#txtUserName').val();
im = new MyIm(window.location.hostname, window.location.port, user);
$('.online').show();
$('.offline').hide();
im.Init();
} //创建一个对象 里面有3个方法,分别为Init:初始化Socket连接、Send:发送消息、Colse:关闭连接
var MyIm = function (path, prot, user_name) {
this.requestPath = 'ws://' + path + ':' + prot + '/tools/Handler.ashx';
this.user_name = user_name;
this.param = '?user_name=' + this.user_name;
this.socekt; }
MyIm.prototype = {
Init: function () {
this.socekt = new WebSocket(this.requestPath + this.param);
this.socekt.onopen = function () {
addSysMessage('连接成功','')
};//连接成功
this.socekt.onmessage = function (result) {
//这里返回的消息为json格式,里面的data为服务器返回的内容
var json = eval('(' + result.data + ')');
if (Number(json.Status) == ) {
addSysMessage(json.Contents,json.Time)
} else {
addMessage(json);
}
};//接收到消息的时候
this.socekt.onclose = function (result) {
addSysMessage('我的连接关闭了','')
}//连接关闭的时候
this.socekt.onerror = function (result) {
addSysMessage('网络发生了错误', '');//当这一步被执行时,close会被自动执行,所以无需主动去执行关闭方法
}//当连接发生错误的时候
},
Send: function (msg) {
//这里可以直接发送消息给服务器,但是为了让服务器好区分我的消息是属于通知还是普通消息还是其他,所以做成了json
//后台获取到json,解析后针对不同的消息类型进行处理
var json = '{"Status":0,"Contents":"'+msg+'","Source":"'+this.user_name+'","Aim":""}';
this.socekt.send(json);
},
Close: function () {
if (this.socekt != null) {
this.socekt.close();
return;
}
}
}
//把通知消息载入到通知列表
function addSysMessage(msg, time) {
if (time.length <= ) {
time = new Date();
}
$('.messageBox').append('<li><p class="time">' + time + '</p><p class=\"message\">' + msg + '</p></li>');
}
//把聊天消息载入到聊天框
function addMessage(json) {
$('.mainBox').append('<li><p class="time">' + json.Time + '</p><p class=\"message\"><span>'+json.Source+'说:</span>' + json.Contents + '</p></li>');
}
//发送消息
function sendMsg() {
var contents = $('#txtContents').val();
im.Send(contents);
}
//关闭连接
function offLine() {
im.Close(); $('.online').hide();
$('.offline').show();
}
</script>
3.客户端HTML:
<div class="mainIm">
<div>
<ul class="mainBox">
</ul>
<ul class="messageBox">
</ul>
</div>
<div class="online" style="display:none;">
<input type="text" id="txtContents" placeholder="输入要发送的内容" />
<input type="button" value="发送" onclick="sendMsg()" /><input type="button" value="断开连接" onclick="offLine()" />
</div>
<div class="offline">
<input type="text" id="txtUserName" placeholder="请输入一个用户名" />
<input type="button" value="连接" onclick="initIm()" />
</div>
</div>
4.最后的效果如下:

三、总结
目前IE并不支持WebSocket,就目前来说,这种方式并不适用于大范围使用。
这里只是实现了简单的聊天室,如果想要一对一,只需找到用户的连接池,向该连接池发送消息即可,如果用户不存在,可以创建一个全局变量离线消息池来储存离线消息。
如有大神发现写的不会的地方,请多多指教!!!
使用Html5下WebSocket搭建简易聊天室的更多相关文章
- php+websocket搭建简易聊天室实践
1.前言 公司游戏里面有个简单的聊天室,了解了之后才知道是node+websocket做的,想想php也来做个简单的聊天室.于是搜集各种资料看文档.找实例自己也写了个简单的聊天室. http连接分为短 ...
- 基于Node.js + WebSocket 的简易聊天室
代码地址如下:http://www.demodashi.com/demo/13282.html Node.js聊天室运行说明 Node.js的本质就是运行在服务端的JavaScript.Node.js ...
- node.js+websocket实现简易聊天室
(文章是从我的个人主页上粘贴过来的,大家也可以访问我的主页 www.iwangzheng.com) websocket提供了一种全双工客户端服务器的异步通信方法,这种通信方法使用ws或者wss协议,可 ...
- node+websocket创建简易聊天室
关于websocket的介绍太多,在这就不一一介绍了,本文主要实现通过websocket创建一个简易聊天室,就是90年代那种聊天室 服务端 1.安装ws模块,uuid模块,ws是websocket模块 ...
- Python开发【笔记】:aiohttp搭建简易聊天室
简易聊天室: 1.入口main.py import logging import jinja2 import aiohttp_jinja2 from aiohttp import web from a ...
- 学习JavaSE TCP/IP协议与搭建简易聊天室
一.TCP/IP协议 1.TCP/IP协议包括TCP.IP和UDP等 2.域名通过dns服务器转换为IP地址 3.局域网可以通过IP或者主机地址寻找到相应的主机 4.TCP是可靠的连接,效率低,且连接 ...
- nodejs+mongoose+websocket搭建xxx聊天室
简介 本文是由nodejs+mongoose+websocket打造的一个即时聊天系统:本来打算开发一个类似于网页QQ类似功能的聊天系统,但是目前只是开发了一个模块功能 --- 类似群聊的,即一对多的 ...
- WebSocket实现简易聊天室
前台页面: <html> <head> <meta http-equiv="Content-Type" content="text/html ...
- websocket搭建的聊天室
在前后端数据交互的时候我们经常使用的是ajax,用的是传统的http协议,而http协议有个致命的缺点,就是请求一结束,连接就断开了, 我们为了保持这个链接的,通常会使用cookie,而自从h5出现w ...
随机推荐
- 老李秘技:loadrunner11.5支持net4.0么?
老李秘技:loadrunner11.5支持net4.0么? LoadRunner12.0以前的版本不支持.NET 4.0,已经证实R&D团队将在下一版本的LoadRunner即LoadRu ...
- hive 锁表问题
报错如下: Unable to acquire IMPLICIT, EXCLUSIVE lock dms@pc_user_msg@month=201611 after 100 attempts. 显示 ...
- emmet(快速开发)的使用
emmet可以帮助您快速编写HTML和CSS代码,从而加速Web前端开发. 比如<html>.<head>.<body>等,现在你只需要1秒钟就可以输入这些标签. ...
- C# Task 源代码(2)
上篇已经讲到Task 的默认的TaskScheduler 为ThreadPoolTaskScheduler. 这时我们回到原来的task 的start方法,在代码最后,调用了 ScheduleAndS ...
- ESLint系列:ESLint入门安装及简单配置
1.eslint需要依赖node.js环境,在配置之前需要安装好node.js; 2.npm install eslint --save-dev 或 npm install eslint --save ...
- 使用Dubbox构架分布式服务
第一部分:Dubbo的背景分析及工作原理 1. Dubbo是什么?Dubbo是一个来自阿里巴巴的开源分布式服务框架,致力于提供高性能和透明化的RPC远程服务调用方案,以及SOA服务治理方案. 简单的说 ...
- Linux--shell脚本之文本处理工具
文本处理工具--grep.sed.awk Bash Shell提供了功能强大的文件处理工具:sed(流编辑器stream editor)和awk,都可使用正则表达式进行模式匹配. 而grep又有助于理 ...
- 浅谈 angular新旧版本问题
一直在学习angularJs,之前用的版本比较老,前些天更新了一下angularJs的版本,然后发现了一些问题,希望和大家分享一下. 在老的版本里控制器直接用函数定义就可以 比如: 在angularJ ...
- JS实现排序
排序算法可以分为内部排序和外部排序.内部排序是数据记录在内存中进行排序,外部排序是因排序的数据很大,一次不能够容纳全部的排序记录,在排序中需要访问外存.常见的内部排序算法有插入排序,选择排序,冒泡排序 ...
- xmlplus 组件设计系列之八 - 分隔框(DividedBox)
分隔框(DividedBox)是一种布局类组件,可以分为两类,其中一类叫水平分隔框(HDividedBox),另一类叫垂直分隔框(VDividedBox).水平分隔框会将其子级分为两列,而垂直分隔框则 ...