ASP.NET MVC 长连接(服务器推)完整实现
1.什么是“服务器推”(百科来一波)?
- 传统模式的 Web 系统以客户端发出请求、服务器端响应的方式工作。这种方式并不能满足很多现实应用的需求,譬如:
- 监控系统:后台硬件热插拔、LED、温度、电压发生变化; 即时通信系统:其它用户登录、发送信息; 即时报价系统:后台数据库内容发生变化; 这些应用都需要服务器能实时地将更新的信息传送到客户端,而无须客户端发出请求。“服务器推”技术在现实应用中有一些解决方案,本文将这些解决方案分为两类:一类需要在浏览器端安装插件,基于套接口传送信息,或是使用 RMI、CORBA 进行远程调用;而另一类则无须浏览器安装任何插件、基于 HTTP 长连接。
- 将“服务器推”应用在 Web 程序中,首先考虑的是如何在功能有限的浏览器端接收、处理信息:
- 客户端如何接收、处理信息,是否需要使用套接口或是使用远程调用。客户端呈现给用户的是 HTML 页面还是 Java applet 或 Flash 窗口。如果使用套接口和远程调用,怎么和 JavaScript 结合修改 HTML 的显示。 客户与服务器端通信的信息格式,采取怎样的出错处理机制。 客户端是否需要支持不同类型的浏览器如 IE、Firefox,是否需要同时支持 Windows 和 Linux 平台。
2.需求
- RFID中间件实现,上百级硬件设备集群管理,大量数据实时上传。
- 中间件管理界面简单大方,带来的是后台异常及日志无法在运行过程中实时跟踪。
- WEB端实时、简单、稳定输出日志。
- 要求输出不同类型的日志,如操作日志、异常日志等。
3.实现
3.1生产者
- 存在的多个浏览器同时打开长连接,也就是存在日志输出一对多的情况,所以生产的回调必须要用到委托链。
- 日志输出不能影响到正常工作流程,所以日志的回调输出必须要用异步加缓存的方式,否则日志将会阻塞。
 直接亮代码(已经完全实现):
public delegate void DebugCallback(Int32 type, String str);                                    // 委托
public class DebugMsg
{
    public int Type { get; set; }
    public String Message { get; set; }
    public DebugMsg(Int32 type, String str)
    {
        this.Type = type; this.Message = str;
    }
}  // 调试消息
public class CallbackManager
{
    #region 单例
    private static CallbackManager instance = null;
    public static CallbackManager Instance                  // 只读属性
    {
        get
        {
            if (instance == null)
            {
                instance = new CallbackManager();
                instance.StartProcess();               // 启用处理线程
            }
            return CallbackManager.instance;
        }
    }
    #endregion
    public static DebugCallback G_D = new DebugCallback(delegate(Int32 type, String str) {
        // System.Diagnostics.Debug.WriteLine("异常类型:" + type + "异常信息:" + str);
    });
    private static int G_D_ClientCount = 0;
    private static int G_D_MAX = 10;                                    // 最大接入客户端数目
    private static object G_D_LOCK = new object();                      // 同步锁
    private static Queue<DebugMsg> QUEUE_BUFF = new Queue<DebugMsg>();  // 调试信息缓存
    private static object QUEUE_BUFF_LOCK = new object();               // 同步锁
    private static Boolean IS_PROCESS = true;
    public CallbackManager() { }
    #region 对外接口
    // 输出调试信息
    public void _D(Int32 type, String str)
    {
        lock (QUEUE_BUFF_LOCK)                                 // 同步操作
        {
            try
            {
                QUEUE_BUFF.Enqueue(new DebugMsg(type, str));
                Monitor.Pulse(QUEUE_BUFF_LOCK);
            }
            catch { };
        }
    }
    // 输出调试信息
    public void _D(String str)
    {
        lock (QUEUE_BUFF_LOCK)                  // 同步操作
        {
            try
            {
                QUEUE_BUFF.Enqueue(new DebugMsg(0, str));
                Monitor.Pulse(QUEUE_BUFF_LOCK);
            }
            catch { };
        }
    }
    // 添加客户端调试信息输出
    public void AddClient(DebugCallback client)
    {
        lock (G_D_LOCK)                  // 同步操作
        {
            try
            {
                if (G_D_ClientCount >= G_D_MAX) return;
                G_D += client;
                G_D_ClientCount++;
                _D("当前调试客户端个数:" + G_D_ClientCount);
            }
            catch { }
        }
    }
    // 删除客户端调试信息输出
    public void RemoveClient(DebugCallback client)
    {
        lock (G_D_LOCK)                  // 同步操作
        {
            try
            {
                G_D -= client;
                G_D_ClientCount--;
                _D("当前调试客户端个数:" + G_D_ClientCount);
            }
            catch { }
        }
    }
    #endregion
    // 处理缓存队列消息
    private void StartProcess()
    {
        ThreadPool.QueueUserWorkItem(new WaitCallback(delegate(object o){
            while (IS_PROCESS)
            {
                lock (QUEUE_BUFF_LOCK)                              // 同步操作
                {
                    if (QUEUE_BUFF.Count > 0)
                    {
                        lock (G_D_LOCK)
                        {
                            try
                            {                                       // 输出异常消息
                                DebugMsg deque = QUEUE_BUFF.Dequeue();
                                G_D(deque.Type, deque.Message);      // 从队列中输出调试消息
                            }
                            catch { }
                        }
                    }
                    else
                    {
                        Monitor.Wait(QUEUE_BUFF_LOCK);
                    }
                }
            }
        }));
    }
3.2消费者
Boolean isOnline = true;
// GET: /System/
public ActionResult Index()
{
    #region 滚动条控制
    Response.Write("<html onclick=\"clearInterval(i_1);\" ondblclick =\"reInterval()\"><head><title>服务器实时监控</title></head>");
    Response.Write("<script type=\"text/javascript\">");
    Response.Write("function scrollWindow() { document.body.scrollTop = document.body.scrollHeight; }");
    Response.Write("function reInterval() { i_1 = setInterval('scrollWindow()', 50); }");
    Response.Write("i_1 = setInterval('scrollWindow()', 50);");
    Response.Write("scrollWindow();");
    Response.Write("</script> ");
    Response.Flush();
    #endregion
    Dictionary<int, string> dicColor = new Dictionary<int, string>()
    {
        {0,"#0000FF"},          // 蓝色
        {1,"#FF3333"},          // 红色
        {2,"#FFFF00"},          // 黄色
        {3,"#FF3EFF"},
        {4,"#0000FF"},
        {5,"#0000FF"}
    };
    DebugCallback callback = new DebugCallback(delegate(int type, string str)
    {
        if (dicColor.ContainsKey(type))
        {
            Response.Write("<span style = 'color:" + dicColor[type] + ";'>" + DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + "--" + str + "</span>  <br />");
        }
        Response.Flush();
    });
    Log.AddCallBack(callback);
    while (isOnline)
    {
        try
        {
            Response.Write("...<br>");
            Response.Flush();
        }
        catch { }
        System.Threading.Thread.Sleep(1000);
        if (!Response.IsClientConnected)            // 连接关闭
        {
            Log.RemoveCallBack(callback);
            Response.Write("</html>");
            break;
        }
    }
    return null;
}
长链接会涉及到Session阻塞问题,详细说明请见:http://www.cnblogs.com/fanqie-liuxiao/p/5702633.html
ASP.NET MVC 长连接(服务器推)完整实现的更多相关文章
- Asp.NET MVC 使用 SignalR 实现推送功能二(Hubs 在线聊天室 获取保存用户信息)
		简单介绍 关于SignalR的简单实用 请参考 Asp.NET MVC 使用 SignalR 实现推送功能一(Hubs 在线聊天室) 在上一篇中,我们只是介绍了简单的消息推送,今天我们来修改一下,实现 ... 
- AngularJS+ASP.NET MVC+SignalR实现消息推送
		原文:AngularJS+ASP.NET MVC+SignalR实现消息推送 背景 OA管理系统中,员工提交申请单,消息实时通知到相关人员及时进行审批,审批之后将结果推送给用户. 技术选择 最开始发现 ... 
- IM推送保障及网络优化详解(二):如何做长连接加推送组合方案
		对于移动APP来说,IM功能正变得越来越重要,它能够创建起人与人之间的连接.社交类产品中,用户与用户之间的沟通可以产生出更好的用户粘性. 在复杂的 Android 生态环境下,多种因素都会造成消息推送 ... 
- 转载——Asp.Net MVC+EF+三层架构的完整搭建过程
		转载http://www.cnblogs.com/zzqvq/p/5816091.html Asp.Net MVC+EF+三层架构的完整搭建过程 架构图: 使用的数据库: 一张公司的员工信息表,测试数 ... 
- Asp.NET MVC 使用 SignalR 实现推送功能一(Hubs 在线聊天室)
		简介 ASP .NET SignalR 是一个ASP .NET 下的类库,可以在ASP .NET 的Web项目中实现实时通信.什么是实时通信的Web呢?就是让客户端(Web页面)和服务器端 ... 
- MarioTCP:一个单机可日30亿的百万并发长连接服务器
		原文:http://blog.csdn.net/everlastinging/article/details/10894493 注:如果用此服务器做变长data的传输,请在业务处理函数中为input ... 
- ASP.NET MVC之读取服务器文件资源的两种方式
		初次认识asp.net mvc时,以为所有文件都需要走一遍路由,然后才能在客户端显示, 所以我首先介绍这一种方式 比如说:我们在服务器上有图片: ~/resource/image/5.jpg 我们就需 ... 
- Asp.NET websocket,Asp.NET MVC 使用 SignalR 实现推送功能一(Hubs 在线聊天室)
		ASP .NET SignalR 是一个ASP .NET 下的类库,可以在ASP .NET 的Web项目中实现实时通信.什么是实时通信的Web呢?就是让客户端(Web页面)和服务器端可以互相通知消息及 ... 
- Asp.Net MVC+EF+三层架构的完整搭建过程
		架构图: 使用的数据库: 一张公司的员工信息表,测试数据 解决方案项目设计: 1.新建一个空白解决方案名称为Company 2.在该解决方案下,新建解决方案文件夹(UI,BLL,DAL,Model) ... 
随机推荐
- 【转-mysql-explain介绍】
			explain显示了MySQL如何使用索引来处理select语句以及连接表.可以帮助选择更好的索引和写出更优化的查询语句. 先解析一条sql语句,看出现什么内容 EXPLAINSELECTs.uid, ... 
- vue render & JSX
			vue在绝大多数使用template是没问题的,但在某些场合下,使用render更适合. 一.render函数 1.createElement 参数 createElement 可接受三个参数 1){ ... 
- [Xcode 实际操作]三、视图控制器-(4)使用UINavigationController导航栏和工具栏
			目录:[Swift]Xcode实际操作 本文将演示如何显示和隐藏导航视图的导航栏和工具栏 打开第一个视图控制器 import UIKit class FirstSubViewController: U ... 
- tomcat与jetty接收请求参数的区别
			[场景] 服务端点对点通知.A服务发起请求B服务,B同步返回接收成功:然后B开始处理逻辑:B处理完成后异步通知给A:A接收请求并处理,同步回写响应给B:完成. [先上代码] 服务端(接收端)代码: i ... 
- 使用java画一张海报
			PS: 没找到合适的海报背景,就随便找了一张,使用技术都是相同的 1. 添加依赖 这俩其实跟本章节的核心技术没有关系,是为了获取QQ昵称和QQ头像而引入的. <!-- jsoup --> ... 
- 与pocket 对接技术文档
			同步每日新增用户接口(kwai 提供) 注释:该接口 每天0点(北京时间)之后 向kwai服务器同步前一天 新增的IMEI号 url:http://m.kwai.com/rest/o/pocket/ ... 
- Python的自增运算与Python变量的浅析
			一.关于Python的自增运算 学了C/C++后再学习Python,不自觉地就打出了自增运算符++,但是发现Python解释器不认识,查了下资料,发现Python中没有这个运算符.这里暂时不探讨自增运 ... 
- Oracle 11g 数据类型
			1. 字符类型 数据类型 长度 说明 CHAR(n BYTE/CHAR) 默认1字节,n值最大为2000 末尾填充空格以达到指定长度,超过最大长度报错.默认指定长度为字节数,字符长度可以从1字 ... 
- POJ1013 Counterfeit Dollar
			题目来源:http://poj.org/problem?id=1013 题目大意:有12枚硬币,其中有一枚假币.所有钱币的外表都一样,所有真币的重量都一样,假币的重量与真币不同,但我们不知道假币的重量 ... 
- groovy使用小记
			下载groovy的sdk, 解压后设置GROOVY_HOME和PATH变量 http://groovy-lang.org/download.html 使用IDEA创建java项目勾选Groovy组件 ... 
