当前位置: 移动技术网 > IT编程>开发语言>c# > C#制作简单的多人在线即时交流聊天室

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

2019年07月18日  | 移动技术网IT编程  | 我要评论

实现网页版的在线聊天室的方法有很多,在没有来到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。状态切换:上线则加入到在线人员集合中,并生成一条上线消息放入到聊天消息集合中,离线则从在线人员集合中移除该人员信息,并生成一条离线消息放入聊天消息集合中;

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

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

 张三:

李四:

小美:

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

如对本文有疑问, 点击进行留言回复!!

相关文章:

验证码:
移动技术网