引言

实现网页版的在线聊天室的方法有很多,在没有来到HTML5之前,常见的有:定时轮询、长连接+长轮询、基于第三方插件(如FLASH的Socket),而如果是HTML5,则比较简单,可以直接使用WebSocket,当然HTML5目前在PC端并没有被所有浏览器支持,所以我的这个聊天室仍是基于长连接+长轮询+原生的JS及AJAX实现的多人在线即时交流聊天室,这个聊天室其实是我上周周末完成的,功能简单,可能有些不足,但可以满足在线即时聊天需求,分享也是给大家提供一个思路,大家可以基于此来实现更好的在线即时聊天工具。

聊天室功能简介:

支持多人进入同一个聊天室聊天;

进入即离线均会自动生成通知信息显示在聊天室中,这样聊天的人们就知道谁进来了谁离开了;

实时显示在线人员表列;

无需数据库支持,全部存在内存中,当然有条件的可以采用分布式缓存或加一个数据库来存,这里演示就是用内存来存了。

下面就开始分享我的代码,由于采用原生的JS及AJAX,所以简单易懂,代码分别WEB前端及服务端(有点废话了)

WEB前端源代码如下:(ChatPage.html)

欢迎进入梦在旅途的网页即时在线大众聊天室 - www.zuowenjun.cn:

当前在线人员

聊天妮称:

输入内容:

代码很简单,并都有注释,在此就不作说明了,如果有疑问欢迎在下方评论。

服务端(ChatHandler.ashx)

using System;

using System.Web;

using System.Collections;

using System.Collections.Generic;

using System.Linq;

using System.Web.Script.Serialization;

using System.Threading;

using System.Collections.Concurrent;

public class ChatHandler : IHttpHandler

{

private class Msg

{

public string name { get; set; }

public string sendtime { get; set; }

public string content { get; set; }

public string readednams { get; set; }

public int readedCount { get; set; }

public string type { get; set; }

}

private static Listmsgs = new List();

private static ReaderWriterLockSlim rwLock = new ReaderWriterLockSlim();

private static object syncObject = new object(),syncObject1 = new object();

private static ListonLineNames = new List();

public void ProcessRequest(HttpContext context)

{

string chatName = context.Request.Form["name"];

string msg = context.Request.Form["msg"];

string actionName = context.Request.Form["action"];

JavaScriptSerializer jsSerializer = new JavaScriptSerializer();

object responseObject = null;

switch (actionName)

{

case "receive":

{

responseObject = GetNewMessages(chatName);

break;

}

case "send":

{

responseObject = SendMessage(chatName, msg, "normal");

break;

}

case "on":

case "off":

{

responseObject = SetChatStatus(chatName, actionName);

break;

}

case "onlines":

{

responseObject = onLineNames;

break;

}

}

context.Response.ContentType = "text/json";

context.Response.Write(jsSerializer.Serialize(responseObject));

}

private object SetChatStatus(string chatName, string status)

{

if (status == "on")

{

if (onLineNames.Exists(s => s == chatName))

{

return new { success = false, info = "该聊天妮称已经存在,请更换一个名称吧!" };

}

lock (syncObject1)

{

onLineNames.Add(chatName);

}

SendMessage(chatName, "大家好,我进入聊天室了!", status);

return new { success = true, info = string.Empty };

}

else

{

lock (syncObject1)

{

onLineNames.Remove(chatName);

}

SendMessage(chatName, "再见,我离开聊天室了!", status);

return new { success = true, info = string.Empty };

}

}

/// /// 获取未读的新消息

///

///

///

private object GetNewMessages(string chatName)

{

//第一种:循环处理

while (true)

{

var newMsgs = msgs.Where(m => m.name != chatName && !(m.readednams ?? "").Contains(chatName)).OrderBy(m => m.sendtime).ToList();

if (newMsgs != null && newMsgs.Count() > 0)

{

lock (syncObject)

{

newMsgs.ForEach((m) =>

{

m.readednams += chatName + ",";

m.readedCount++;

});

int chatNameCount = onLineNames.Count();

msgs.RemoveAll(m => m.readedCount >= chatNameCount);

}

return new { success = true, msgs = newMsgs };

}

Thread.Sleep(1000);

}

//第二种方法,采用自旋锁

//ListnewMsgs = null;

//SpinWait.SpinUntil(() =>

//{

// newMsgs = msgs.Where(m => m.name != chatName && !(m.readednams ?? "").Contains(chatName)).OrderBy(m => m.sendtime).ToList();

// return newMsgs.Count() > 0;

//}, -1);

//rwLock.EnterWriteLock();

//newMsgs.ForEach(m =>

//{

// m.readednams += chatName + ",";

// m.readedCount++;

//});

//rwLock.ExitWriteLock();

//return new { success = true, msgs = newMsgs };

}

/// ///

///

///

///

///

private object SendMessage(string chatName, string msg, string type)

{

var newMsg = new Msg() { name = chatName, sendtime = DateTime.Now.ToString("yyyy/MM/dd HH:mm"), content =HttpContext.Current.Server.HtmlEncode(msg), readednams = null, type = type };

//rwLock.EnterWriteLock();

lock (syncObject)

{

msgs.Add(newMsg);

}

//rwLock.ExitWriteLock();

return new { success = true, msgs = new[] { newMsg } };

}

public bool IsReusable

{

get

{

return false;

}

}

}

代码也相对简单,实现原理主要是:

聊天消息:循环获取未读的消息,在取出读的消息同时,将其标识为已读,全部已读的消息则删除;--我这里采用了两种方法,第二种方法被注释掉了,大家可以取消注释试试,也是不错的,比第一种更直观,建议使用;

发送消息:实例化一个消息实例并加入到聊天消息集合中;

状态切换:上线则加入到在线人员集合中,并生成一条上线消息放入到聊天消息集合中,离线则从在线人员集合中移除该人员信息,并生成一条离线消息放入聊天消息集合中;

注意事项,由于采用了全局静态集合,所以线程同步比较重要。

最终的实现效果展示如下:

如果觉得不错的话,希望给我一个关注吧。

网页版聊天服务器,网页版在线聊天室相关推荐

  1. 基于websocket的网页实时消息推送与在线聊天(上篇)

    文章目录 @[toc] 基于websocket的网页实时消息推送与在线聊天(上篇) "使用dwebsocket在django中实现websocket" websocket原理图 d ...

  2. 高仿Telegram IM Chat 聊天软件 即时通讯 在线聊天加社群 纯静态UI架构介绍

    高仿Telegram IM Chat 聊天软件 即时通讯 在线聊天加社群 纯静态UI 全端覆盖!苹果手机端.安卓手机端.网页版.电脑PC版.无论哪个平台,都能畅通无阻! 性价比高!功能全面而价格低廉! ...

  3. ftp服务器网页空白,ftp服务器网页空白

    ftp服务器网页空白 内容精选 换一换 BINDIR为Cntlm的启动目录,/opt/cntlm/etc为Cntlm的配置文件目录,/opt/cntlm/share/man为Cntlm的帮助指南目录. ...

  4. 网页的云服务器,网页与云服务器

    网页与云服务器 内容精选 换一换 云平台支持修改主网卡的私有IP地址,具体操作请参见本节内容.如需修改扩展网卡的私有IP地址,请删除网卡,并挂载新网卡.云服务器已关机.如果网卡绑定了虚拟IP或者DNA ...

  5. (十三)nodejs循序渐进-高性能游戏服务器框架pomelo之扩展聊天服务器为机器人自动聊天

    聊天服务器扩展 大家在上一篇文章里相信已经学会了pomelo框架的基本用法了,那么我们在上一篇文章的代码基础上继续扩展,丰富系统,另外也熟悉下他的更多的用法,这一节我将扩展它:增加一个机器人自动聊天的 ...

  6. lol好友列表显示聊天服务器断开,lol聊天服务器断开 英雄联盟聊天服务器连不上解决办法...

    亲们在玩英雄联盟的时候经常会遇到一个情况:右侧的LOL聊天服务器断开:尤其大家在跟朋友开黑的时候遇到这个情况无法拉好友加入游戏是非常恼火的,本文就聊天服务器连不上这个问题为大家详细解答,只需一分钟包教 ...

  7. socket.io html5 聊天,socket.io实现在线聊天页面

    一.要点分析 (1) 对于socket.io,它是基于事件响应的socket,可以进行长时间的消息传递.其服务端使用的方法主要不过就是两个,on()和emit() io.on('connetcion' ...

  8. 登录网页无法连接服务器,网页游戏无法登录全面分析

    根据最近一些玩家的反应,玩家正在进行游戏的时候突然掉线.卡加载进度条.黑屏.白屏.服务器不允许链接等,这些都是网页游戏的最为常见问题.下面由我介绍四种方法给各位玩家,请您们遇到问题可以按照以上方法尝试 ...

  9. 服务器网页篡改,网站服务器网页防篡改系统

    网页防篡改系统设计理念 网页防篡改系统在站点采用了两种防范方法,实现对静态区域文件和动态区域文件的保护.动态区域文件保护主要是在站点架设Web防火墙,通过设定关键字.IP.时间过滤规则,对扫描,非法访 ...

  10. php ws聊天,php+websocket实现在线聊天室

    熟悉web开发的朋友明白,平时我们与服务端的数据请求都是基于HTTP协议,而HTTP协议,通信请求只能由客户端发起,服务端对请求做出应答处理.也就是说服务端不能主动向我们推送数据,而像即时通讯这类应用 ...

最新文章

  1. Deep Learning部署TVM Golang运行时Runtime
  2. liunx 学习笔记-wzq
  3. C#.NET 通用权限管理系统组件 大数据多表分页获取部分列的参考方法
  4. mysql explain详解_数据库mysql(1)——B+TREE索引原理
  5. git中的gitgnore是什么? 码云gitee
  6. angular i18n 国际化 多语言
  7. bzoj#4555. [Tjoi2016Heoi2016]求和
  8. Hadoop2.x环境搭建
  9. IOS::.a的生成,以及模拟器和真机.a 的合并
  10. Redis(二)原理以及主从复制的实现
  11. java script应用领域_JavaScript的应用
  12. matlab在遥感数字图像处理方面的应用
  13. c语言不能正常输出128以上的ascii字符,C语言输出ASCII 219无法显示方格
  14. 基于python的猜大小游戏
  15. 大学计算机第一课知识点,大学计算机基础第一课
  16. Beyong Compare4过期解决办法
  17. JAVA导出exls时报oom,Poi导出产生OOM解决方案
  18. MySQL约束和表的复杂查询操作
  19. Hadoop——MapReduce相关eclipse配置及Api调用(图文超详细版)(内含遇到错误的解决方法)
  20. 微信设置“种草昵称”,个性有意思,来看看!

热门文章

  1. Mybatis中文文档下载地址分享
  2. 计算机维护的工作周记,计算机维护实习周记
  3. 智能优化算法:人工水母搜索算法 -附代码
  4. robot framework接口自动化测试post请求
  5. DroidCam通过数据线调用手机摄像头的方法二
  6. masm5安装教程_MASM使用方法及版本号
  7. FlashFXP 5安装、远程访问服务器、以及修改密码
  8. 100kW以上 中高频感应加热电源 双DSP数字式IGBT控制板
  9. 谷歌地球(google earth)中文专业版官方下载
  10. 关于金融行业网络安全解决方案,您了解多少?