C#制作简单的多人在线即时交流聊天室

作者:hebedich 时间:2023-02-23 12:12:23 

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

聊天室功能简介:

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

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

3。实时显示在线人员表列;

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

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

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


<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
 <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
 <title></title>
 <style type="text/css">
   html, body {
     margin: 0px;
     padding: 0px;
     width: 100%;
     height: 100%;
     background-color: #f8f7f7;
     font-family: arial,sans-serif;
   }

#layouttable {
     margin:0px;
     padding:0px;
     width:100%;
     height:100%;
     border:2px solid green;
     border-collapse:collapse;
     min-width:800px;
   }

#layouttable td {
       border: 1px solid green;
     }

.h100p {
     height:100%;
   }

.midtr{height:auto;}
     .midtr tr td {
       height: 100%;
     }

#chatmsgbox, #chatonlinebox {
     background-color:white;
     overflow-x: hidden;
     overflow-y: auto;
     overflow-wrap: break-word;
     height: 100%;
   }

#chatonlinebox {
     background-color:#f5d0a8;
   }

.rc, .sd {
     overflow:hidden;
   }

.rc p {
     float: left;
     color: green;
   }
     .sd p {
       float: right;
       color: orange;
     }
 </style>

</head>
<body>
 <table id="layouttable">
   <colgroup>
     <col style="width:auto" />
     <col style="width: 200px;" />
   </colgroup>
   <tr style="height:30px; background-color:lightblue;color:yellow;">
     <td>
       欢迎进入梦在旅途的网页即时在线大众聊天室 - www.zuowenjun.cn:
     </td>
     <td>
       当前在线人员
     </td>
   </tr>
   <tr style="height:auto;" id="midtr">
     <td>
       <div id="chatmsgbox">
       </div>
     </td>
     <td>
       <div id="chatonlinebox">
         <ul id="chatnames"></ul>
       </div>
     </td>
   </tr>
   <tr style="height:50px;">
     <td colspan="2">
       <label for="name">聊天妮称:</label>
       <input type="text" id="name" style="width:80px;" />
       <input type="button" id="btnsavename" value="确认进入" />
       <label for="msg">输入内容:</label>
       <input type="text" id="msg" style="width:400px;" />
       <input type="button" id="btnSend" value="发送消息" disabled="disabled" />
     </td>
   </tr>
 </table>
 <script type="text/javascript">
   var chatName = null;
   var oChatmsgbox, oMsg, oChatnames;
   var ajaxforSend, ajaxforRecv;

//页面加载初始化
   window.onload = function () {
     document.getElementById("btnsavename").onclick = function () {
       this.disabled = true;
       var oName = document.getElementById("name");
       oName.readOnly = true;
       document.getElementById("btnSend").disabled = false;
       //receiveMsg();
       setChatStatus(oName.value,"on");
     }

document.getElementById("btnSend").onclick = function () {
       sendMsg(oMsg.value);
     };

//init
     oChatmsgbox = document.getElementById("chatmsgbox");
     oMsg = document.getElementById("msg");
     oChatnames = document.getElementById("chatnames");
     ajaxforSend = getAjaxObject();
     ajaxforRecv = getAjaxObject();
   }

//离开时提醒
   window.onbeforeunload = function () {
     event.returnValue = "您确定要退出聊天室吗?";
   }

//关闭时离线
   window.onunload = function () {
     setChatStatus(chatName, "off");
   }

//设置聊天状态:在线 OR 离线
   function setChatStatus(name, status) {
     callAjax(getAjaxObject(), "action=" + status + "&name=" + name, function (rs) {
       if (!rs.success) {
         alert(rs.info);
         return;
       }
       if (status == "on") {
         chatName = document.getElementById("name").value;
         setTimeout("receiveMsg()",500);
       }
       loadOnlineChatNames();
     });
   }

//加载在线人员名称列表
   function loadOnlineChatNames(){
     callAjax(getAjaxObject(), "action=onlines", function (rs) {
       var lis = "";
       for(var i=0;i<rs.length;i++)
       {
         lis += "<li>"+ rs[i] +"</li>";
       }
       oChatnames.innerHTML = lis;
     });
   }

//接收消息列表
   function receiveMsg() {
     callAjax(ajaxforRecv, "action=receive&name=" + chatName, function (rs) {
       if (rs.success) {
         showChatMsgs(rs.msgs, "rc");
       }
       setTimeout("receiveMsg()", 500);
     });
   }
   //发送消息
   function sendMsg(msg) {
     callAjax(ajaxforSend, "action=send&name=" + chatName + "&msg=" + escape(msg), function (rs) {
       if (rs.success) {
         showChatMsgs(rs.msgs, "sd");
         oMsg.value = null;
         //alert("发送成功!");
       }
     });
   }

//显示消息
   function showChatMsgs(msgs, cssClass) {
     var loadonline = false;
     for (var i = 0; i < msgs.length; i++) {
       var msg = msgs[i];
       oChatmsgbox.innerHTML += "<div class='" + cssClass + "'><p>[" + msg.name + "] - " + msg.sendtime + " 说:<br/>" + msg.content + "</p></div>";
       if (msg.type == "on" || msg.type == "off")
       {
         loadonline = true;
       }
     }
     if (loadonline)
     {
       loadOnlineChatNames();
     }
   }

//调用AJAX
   function callAjax(ajax, param, callback) {

ajax.open("post", "ChatHandler.ashx", true);
     ajax.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
     ajax.onreadystatechange = function () {
       if (ajax.readyState == 4 && ajax.status == 200) {
         var json = eval("(" + ajax.responseText + ")");
         callback(json);
       }
     };
     ajax.send(param);
   }

//获取AJAX对象(XMLHttpRequest)
   function getAjaxObject() {
     var xmlhttp;
     if (window.XMLHttpRequest) {// code for IE7+, Firefox, Chrome, Opera, Safari
       xmlhttp = new XMLHttpRequest();
     }
     else {// code for IE6, IE5
       xmlhttp = new ActiveXObject("Microsoft.XMLHTTP");
     }
     return xmlhttp;
   }

</script>
</body>
</html>

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

服务端(ChatHandler.ashx)


<%@ WebHandler Language="C#" Class="ChatHandler" %>

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 List<Msg> msgs = new List<Msg>();
 private static ReaderWriterLockSlim rwLock = new ReaderWriterLockSlim();
 private static object syncObject = new object(),syncObject1 = new object();
 private static List<string> onLineNames = new List<string>();

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 };
   }
 }

/// <summary>
 /// 获取未读的新消息
 /// </summary>
 /// <param name="chatName"></param>
 /// <returns></returns>
 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);
   }

//第二种方法,采用自旋锁
   //List<Msg> newMsgs = 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 };
 }

/// <summary>
 ///
 /// </summary>
 /// <param name="chatName"></param>
 /// <param name="msg"></param>
 /// <returns></returns>
 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。聊天消息:循环获取未读的消息,在取出读的消息同时,将其标识为已读,全部已读的消息则删除;--我这里采用了两种方法,第二种方法被注释掉了,大家可以取消注释试试,也是不错的,比第一种更直观,建议使用;

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

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

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

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

 张三:

C#制作简单的多人在线即时交流聊天室

李四:

C#制作简单的多人在线即时交流聊天室

小美:

C#制作简单的多人在线即时交流聊天室

如果觉得不错的话,给个推荐吧,你的支持是推动我不断前进的动力及写作的源泉,我一直坚持:知识在于分享,分享的同时自己也在成长,希望与大家共同成长,谢谢!

标签:C#,聊天室
0
投稿

猜你喜欢

  • Android EventBus 3.0.0 使用总结(必看篇)

    2023-09-06 06:32:41
  • java微信公众号开发第一步 公众号接入和access_token管理

    2022-04-21 23:06:41
  • Java中五种不同方法的创建对象

    2021-07-25 01:11:37
  • java poi设置生成的word的图片为上下型环绕以及其位置的实现

    2023-05-23 03:12:08
  • Java使用DateTimeFormatter格式化输入的日期时间

    2023-03-09 04:52:38
  • SpringCloud OpenFeign超详细讲解模板化远程通信的实现

    2023-04-27 06:43:29
  • c#异步操作async await状态机的总结(推荐)

    2021-08-22 07:13:12
  • C# 手动/自动保存图片的实例代码

    2023-08-06 09:36:35
  • 浅谈Springboot之于Spring的优势

    2022-01-08 02:35:29
  • 基于Android本地代码生成器详解

    2022-09-15 02:52:26
  • Kotlin 协程与挂起函数及suspend关键字深入理解

    2023-09-19 01:04:25
  • Java 1.0和Java 1.1 的IO类的比较

    2023-10-07 10:48:39
  • C#检测是否有u盘插入的方法

    2023-11-19 13:11:00
  • maven+阿里云创建国内镜像的中央仓库(亲测可用)

    2023-05-12 22:38:58
  • WebFlux 服务编排使用优势详解

    2021-11-27 07:48:45
  • Winform让DataGridView左侧显示图片

    2021-09-24 03:50:38
  • Java 语言实现清除带 html 标签的内容方法

    2021-07-29 23:47:43
  • android实现简单的画画板实例代码

    2023-02-21 05:46:02
  • java中类加载与双亲委派机制详解

    2023-12-07 06:34:53
  • Java线程池实现原理总结

    2023-04-15 02:37:11
  • asp之家 软件编程 m.aspxhome.com