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

2025-12-13 0 859

实现网页版的在线聊天室的方法有很多,在没有来到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#制作简单的多人在线即时交流聊天室

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

作者:hebedich

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

申明:本文由第三方发布,内容仅代表作者观点,与本网站无关。对本文以及其中全部或者部分内容的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。本网发布或转载文章出于传递更多信息之目的,并不意味着赞同其观点或证实其描述,也不代表本网对其真实性负责。

左子网 编程相关 C#制作简单的多人在线即时交流聊天室 https://www.zuozi.net/36567.html

常见问题
  • 1、自动:拍下后,点击(下载)链接即可下载;2、手动:拍下后,联系卖家发放即可或者联系官方找开发者发货。
查看详情
  • 1、源码默认交易周期:手动发货商品为1-3天,并且用户付款金额将会进入平台担保直到交易完成或者3-7天即可发放,如遇纠纷无限期延长收款金额直至纠纷解决或者退款!;
查看详情
  • 1、描述:源码描述(含标题)与实际源码不一致的(例:货不对板); 2、演示:有演示站时,与实际源码小于95%一致的(但描述中有”不保证完全一样、有变化的可能性”类似显著声明的除外); 3、发货:不发货可无理由退款; 4、安装:免费提供安装服务的源码但卖家不履行的; 5、收费:价格虚标,额外收取其他费用的(但描述中有显著声明或双方交易前有商定的除外); 6、其他:如质量方面的硬性常规问题BUG等。 注:经核实符合上述任一,均支持退款,但卖家予以积极解决问题则除外。
查看详情
  • 1、左子会对双方交易的过程及交易商品的快照进行永久存档,以确保交易的真实、有效、安全! 2、左子无法对如“永久包更新”、“永久技术支持”等类似交易之后的商家承诺做担保,请买家自行鉴别; 3、在源码同时有网站演示与图片演示,且站演与图演不一致时,默认按图演作为纠纷评判依据(特别声明或有商定除外); 4、在没有”无任何正当退款依据”的前提下,商品写有”一旦售出,概不支持退款”等类似的声明,视为无效声明; 5、在未拍下前,双方在QQ上所商定的交易内容,亦可成为纠纷评判依据(商定与描述冲突时,商定为准); 6、因聊天记录可作为纠纷评判依据,故双方联系时,只与对方在左子上所留的QQ、手机号沟通,以防对方不承认自我承诺。 7、虽然交易产生纠纷的几率很小,但一定要保留如聊天记录、手机短信等这样的重要信息,以防产生纠纷时便于左子介入快速处理。
查看详情

相关文章

猜你喜欢
发表评论
暂无评论
官方客服团队

为您解决烦忧 - 24小时在线 专业服务