首页 > 编程 > C# > 正文

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

2019-10-29 21:35:42
字体:
来源:转载
供稿:网友

这篇文章主要介绍了C#制作简单的多人在线即时交流聊天室的相关资料,需要的朋友可以参考下

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

聊天室功能简介:

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

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

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

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

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

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

 

  1. <!DOCTYPE html> 
  2. <html xmlns="http://www.w3.org/1999/xhtml"
  3. <head> 
  4. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
  5. <title></title> 
  6. <style type="text/css"
  7. html, body { 
  8. margin: 0px; 
  9. padding: 0px; 
  10. width: 100%; 
  11. height: 100%; 
  12. background-color: #f8f7f7; 
  13. font-family: arial,sans-serif; 
  14.  
  15. #layouttable { 
  16. margin:0px; 
  17. padding:0px; 
  18. width:100%; 
  19. height:100%; 
  20. border:2px solid green; 
  21. border-collapse:collapse; 
  22. min-width:800px; 
  23.  
  24. #layouttable td { 
  25. border: 1px solid green; 
  26.  
  27. .h100p { 
  28. height:100%; 
  29.  
  30. .midtr{height:auto;} 
  31. .midtr tr td { 
  32. height: 100%; 
  33.  
  34. #chatmsgbox, #chatonlinebox { 
  35. background-color:white; 
  36. overflow-x: hidden; 
  37. overflow-y: auto; 
  38. overflow-wrap: break-word; 
  39. height: 100%; 
  40.  
  41. #chatonlinebox { 
  42. background-color:#f5d0a8; 
  43.  
  44. .rc, .sd { 
  45. overflow:hidden; 
  46.  
  47. .rc p { 
  48. float: left; 
  49. color: green; 
  50. .sd p { 
  51. float: right; 
  52. color: orange; 
  53. </style> 
  54.  
  55. </head> 
  56. <body> 
  57. <table id="layouttable"
  58. <colgroup> 
  59. <col style="width:auto" /> 
  60. <col style="width: 200px;" /> 
  61. </colgroup> 
  62. <tr style="height:30px; background-color:lightblue;color:yellow;"
  63. <td> 
  64. 欢迎进入梦在旅途的网页即时在线大众聊天室 - www.zuowenjun.cn: 
  65. </td> 
  66. <td> 
  67. 当前在线人员 
  68. </td> 
  69. </tr> 
  70. <tr style="height:auto;" id="midtr"
  71. <td> 
  72. <div id="chatmsgbox"
  73. </div> 
  74. </td> 
  75. <td> 
  76. <div id="chatonlinebox"
  77. <ul id="chatnames"></ul> 
  78. </div> 
  79. </td> 
  80. </tr> 
  81. <tr style="height:50px;"
  82. <td colspan="2"
  83. <label for="name">聊天妮称:</label> 
  84. <input type="text" id="name" style="width:80px;" /> 
  85. <input type="button" id="btnsavename" value="确认进入" /> 
  86. <label for="msg">输入内容:</label> 
  87. <input type="text" id="msg" style="width:400px;" /> 
  88. <input type="button" id="btnSend" value="发送消息" disabled="disabled" /> 
  89. </td> 
  90. </tr> 
  91. </table> 
  92. <script type="text/javascript"
  93. var chatName = null
  94. var oChatmsgbox, oMsg, oChatnames; 
  95. var ajaxforSend, ajaxforRecv; 
  96.  
  97. //页面加载初始化 
  98. window.onload = function () { 
  99. document.getElementById("btnsavename").onclick = function () { 
  100. this.disabled = true
  101. var oName = document.getElementById("name"); 
  102. oName.readOnly = true
  103. document.getElementById("btnSend").disabled = false
  104. //receiveMsg(); 
  105. setChatStatus(oName.value,"on"); 
  106.  
  107. document.getElementById("btnSend").onclick = function () { 
  108. sendMsg(oMsg.value); 
  109. }; 
  110.  
  111. //init 
  112. oChatmsgbox = document.getElementById("chatmsgbox"); 
  113. oMsg = document.getElementById("msg"); 
  114. oChatnames = document.getElementById("chatnames"); 
  115. ajaxforSend = getAjaxObject(); 
  116. ajaxforRecv = getAjaxObject(); 
  117.  
  118. //离开时提醒 
  119. window.onbeforeunload = function () { 
  120. event.returnValue = "您确定要退出聊天室吗?"
  121.  
  122. //关闭时离线 
  123. window.onunload = function () { 
  124. setChatStatus(chatName, "off"); 
  125.  
  126. //设置聊天状态:在线 OR 离线 
  127. function setChatStatus(name, status) { 
  128. callAjax(getAjaxObject(), "action=" + status + "&name=" + name, function (rs) { 
  129. if (!rs.success) { 
  130. alert(rs.info); 
  131. return
  132. if (status == "on") { 
  133. chatName = document.getElementById("name").value; 
  134. setTimeout("receiveMsg()",500); 
  135. loadOnlineChatNames(); 
  136. }); 
  137.  
  138. //加载在线人员名称列表 
  139. function loadOnlineChatNames(){ 
  140. callAjax(getAjaxObject(), "action=onlines"function (rs) { 
  141. var lis = ""
  142. for(var i=0;i<rs.length;i++) 
  143. lis += "<li>"+ rs[i] +"</li>"
  144. oChatnames.innerHTML = lis; 
  145. }); 
  146.  
  147. //接收消息列表 
  148. function receiveMsg() { 
  149. callAjax(ajaxforRecv, "action=receive&name=" + chatName, function (rs) { 
  150. if (rs.success) { 
  151. showChatMsgs(rs.msgs, "rc"); 
  152. setTimeout("receiveMsg()", 500); 
  153. }); 
  154. //发送消息 
  155. function sendMsg(msg) { 
  156. callAjax(ajaxforSend, "action=send&name=" + chatName + "&msg=" + escape(msg), function (rs) { 
  157. if (rs.success) { 
  158. showChatMsgs(rs.msgs, "sd"); 
  159. oMsg.value = null
  160. //alert("发送成功!"); 
  161. }); 
  162.  
  163. //显示消息 
  164. function showChatMsgs(msgs, cssClass) { 
  165. var loadonline = false
  166. for (var i = 0; i < msgs.length; i++) { 
  167. var msg = msgs[i]; 
  168. oChatmsgbox.innerHTML += "<div class='" + cssClass + "'><p>[" + msg.name + "] - " + msg.sendtime + " 说:<br/>" + msg.content + "</p></div>"
  169. if (msg.type == "on" || msg.type == "off"
  170. loadonline = true
  171. if (loadonline) 
  172. loadOnlineChatNames(); 
  173.  
  174. //调用AJAX 
  175. function callAjax(ajax, param, callback) { 
  176.  
  177. ajax.open("post""ChatHandler.ashx"true); 
  178. ajax.setRequestHeader("Content-type""application/x-www-form-urlencoded"); 
  179. ajax.onreadystatechange = function () { 
  180. if (ajax.readyState == 4 && ajax.status == 200) { 
  181. var json = eval("(" + ajax.responseText + ")"); 
  182. callback(json); 
  183. }; 
  184. ajax.send(param); 
  185.  
  186. //获取AJAX对象(XMLHttpRequest) 
  187. function getAjaxObject() { 
  188. var xmlhttp; 
  189. if (window.XMLHttpRequest) {// code for IE7+, Firefox, Chrome, Opera, Safari 
  190. xmlhttp = new XMLHttpRequest(); 
  191. else {// code for IE6, IE5 
  192. xmlhttp = new ActiveXObject("Microsoft.XMLHTTP"); 
  193. return xmlhttp; 
  194.  
  195. </script> 
  196. </body> 
  197. </html> 

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

服务端(ChatHandler.ashx)

 

  1. <%@ WebHandler Language="C#" Class="ChatHandler" %> 
  2.  
  3. using System; 
  4. using System.Web; 
  5. using System.Collections; 
  6. using System.Collections.Generic; 
  7. using System.Linq; 
  8. using System.Web.Script.Serialization; 
  9. using System.Threading; 
  10. using System.Collections.Concurrent; 
  11.  
  12. public class ChatHandler : IHttpHandler 
  13.  
  14. private class Msg 
  15. public string name { get; set; } 
  16. public string sendtime { get; set; } 
  17. public string content { get; set; } 
  18. public string readednams { get; set; } 
  19. public int readedCount { get; set; } 
  20. public string type { get; set; } 
  21.  
  22. private static List<Msg> msgs = new List<Msg>(); 
  23. private static ReaderWriterLockSlim rwLock = new ReaderWriterLockSlim(); 
  24. private static object syncObject = new object(),syncObject1 = new object(); 
  25. private static List<string> onLineNames = new List<string>(); 
  26.  
  27. public void ProcessRequest(HttpContext context) 
  28. string chatName = context.Request.Form["name"]; 
  29. string msg = context.Request.Form["msg"]; 
  30. string actionName = context.Request.Form["action"]; 
  31. JavaScriptSerializer jsSerializer = new JavaScriptSerializer(); 
  32.  
  33. object responseObject = null
  34.  
  35. switch (actionName) 
  36. case "receive"
  37. responseObject = GetNewMessages(chatName); 
  38. break
  39. case "send"
  40. responseObject = SendMessage(chatName, msg, "normal"); 
  41. break
  42. case "on"
  43. case "off"
  44. responseObject = SetChatStatus(chatName, actionName); 
  45. break
  46. case "onlines"
  47. responseObject = onLineNames; 
  48. break
  49.  
  50. context.Response.ContentType = "text/json"
  51. context.Response.Write(jsSerializer.Serialize(responseObject)); 
  52.  
  53.  
  54. private object SetChatStatus(string chatName, string status) 
  55. if (status == "on"
  56. if (onLineNames.Exists(s => s == chatName)) 
  57. return new { success = false, info = "该聊天妮称已经存在,请更换一个名称吧!" }; 
  58. lock (syncObject1) 
  59. onLineNames.Add(chatName); 
  60. SendMessage(chatName, "大家好,我进入聊天室了!", status); 
  61. return new { success = true, info = string.Empty }; 
  62. else 
  63. lock (syncObject1) 
  64. onLineNames.Remove(chatName); 
  65. SendMessage(chatName, "再见,我离开聊天室了!", status); 
  66. return new { success = true, info = string.Empty }; 
  67.  
  68. /// <summary> 
  69. /// 获取未读的新消息 
  70. /// </summary> 
  71. /// <param name="chatName"></param> 
  72. /// <returns></returns> 
  73. private object GetNewMessages(string chatName) 
  74. //第一种:循环处理 
  75. while (true
  76.  
  77. var newMsgs = msgs.Where(m => m.name != chatName && !(m.readednams ?? "").Contains(chatName)).OrderBy(m => m.sendtime).ToList(); 
  78. if (newMsgs != null && newMsgs.Count() > 0) 
  79. lock (syncObject) 
  80. newMsgs.ForEach((m) => 
  81. m.readednams += chatName + ","
  82. m.readedCount++; 
  83. }); 
  84. int chatNameCount = onLineNames.Count(); 
  85. msgs.RemoveAll(m => m.readedCount >= chatNameCount); 
  86.  
  87. return new { success = true, msgs = newMsgs }; 
  88.  
  89. Thread.Sleep(1000); 
  90.  
  91.  
  92. //第二种方法,采用自旋锁 
  93. //List<Msg> newMsgs = null; 
  94. //SpinWait.SpinUntil(() => 
  95. //{ 
  96. // newMsgs = msgs.Where(m => m.name != chatName && !(m.readednams ?? "").Contains(chatName)).OrderBy(m => m.sendtime).ToList(); 
  97. // return newMsgs.Count() > 0; 
  98. //}, -1); 
  99.  
  100. //rwLock.EnterWriteLock(); 
  101. //newMsgs.ForEach(m => 
  102. //{ 
  103. // m.readednams += chatName + ","; 
  104. // m.readedCount++; 
  105. //}); 
  106. //rwLock.ExitWriteLock(); 
  107. //return new { success = true, msgs = newMsgs }; 
  108.  
  109. /// <summary> 
  110. /// 
  111. /// </summary> 
  112. /// <param name="chatName"></param> 
  113. /// <param name="msg"></param> 
  114. /// <returns></returns> 
  115. private object SendMessage(string chatName, string msg, string type) 
  116. 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 }; 
  117. //rwLock.EnterWriteLock(); 
  118. lock (syncObject) 
  119. msgs.Add(newMsg); 
  120. //rwLock.ExitWriteLock(); 
  121. return new { success = true, msgs = new[] { newMsg } }; 
  122.  
  123.  
  124.  
  125. public bool IsReusable 
  126. get 
  127. return false
  128.  

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

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

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

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

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

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

张三:

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

李四:

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

小美:

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

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


注:相关教程知识阅读请移步到c#教程频道。
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表