1.什么是“服务器推”(百科来一波)?


  • 传统模式的 Web 系统以客户端发出请求、服务器端响应的方式工作。这种方式并不能满足很多现实应用的需求,譬如:
  • 监控系统:后台硬件热插拔、LED、温度、电压发生变化; 即时通信系统:其它用户登录、发送信息; 即时报价系统:后台数据库内容发生变化; 这些应用都需要服务器能实时地将更新的信息传送到客户端,而无须客户端发出请求。“服务器推”技术在现实应用中有一些解决方案,本文将这些解决方案分为两类:一类需要在浏览器端安装插件,基于套接口传送信息,或是使用 RMI、CORBA 进行远程调用;而另一类则无须浏览器安装任何插件、基于 HTTP 长连接。
  • 将“服务器推”应用在 Web 程序中,首先考虑的是如何在功能有限的浏览器端接收、处理信息:
  • 客户端如何接收、处理信息,是否需要使用套接口或是使用远程调用。客户端呈现给用户的是 HTML 页面还是 Java applet 或 Flash 窗口。如果使用套接口和远程调用,怎么和 JavaScript 结合修改 HTML 的显示。 客户与服务器端通信的信息格式,采取怎样的出错处理机制。 客户端是否需要支持不同类型的浏览器如 IE、Firefox,是否需要同时支持 Windows 和 Linux 平台。

2.需求


  1. RFID中间件实现,上百级硬件设备集群管理,大量数据实时上传。
  2. 中间件管理界面简单大方,带来的是后台异常及日志无法在运行过程中实时跟踪。
  3. WEB端实时、简单、稳定输出日志。
  4. 要求输出不同类型的日志,如操作日志、异常日志等。

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 长连接(服务器推)完整实现的更多相关文章

  1. Asp.NET MVC 使用 SignalR 实现推送功能二(Hubs 在线聊天室 获取保存用户信息)

    简单介绍 关于SignalR的简单实用 请参考 Asp.NET MVC 使用 SignalR 实现推送功能一(Hubs 在线聊天室) 在上一篇中,我们只是介绍了简单的消息推送,今天我们来修改一下,实现 ...

  2. AngularJS+ASP.NET MVC+SignalR实现消息推送

    原文:AngularJS+ASP.NET MVC+SignalR实现消息推送 背景 OA管理系统中,员工提交申请单,消息实时通知到相关人员及时进行审批,审批之后将结果推送给用户. 技术选择 最开始发现 ...

  3. IM推送保障及网络优化详解(二):如何做长连接加推送组合方案

    对于移动APP来说,IM功能正变得越来越重要,它能够创建起人与人之间的连接.社交类产品中,用户与用户之间的沟通可以产生出更好的用户粘性. 在复杂的 Android 生态环境下,多种因素都会造成消息推送 ...

  4. 转载——Asp.Net MVC+EF+三层架构的完整搭建过程

    转载http://www.cnblogs.com/zzqvq/p/5816091.html Asp.Net MVC+EF+三层架构的完整搭建过程 架构图: 使用的数据库: 一张公司的员工信息表,测试数 ...

  5. Asp.NET MVC 使用 SignalR 实现推送功能一(Hubs 在线聊天室)

    简介       ASP .NET SignalR 是一个ASP .NET 下的类库,可以在ASP .NET 的Web项目中实现实时通信.什么是实时通信的Web呢?就是让客户端(Web页面)和服务器端 ...

  6. MarioTCP:一个单机可日30亿的百万并发长连接服务器

    原文:http://blog.csdn.net/everlastinging/article/details/10894493 注:如果用此服务器做变长data的传输,请在业务处理函数中为input ...

  7. ASP.NET MVC之读取服务器文件资源的两种方式

    初次认识asp.net mvc时,以为所有文件都需要走一遍路由,然后才能在客户端显示, 所以我首先介绍这一种方式 比如说:我们在服务器上有图片: ~/resource/image/5.jpg 我们就需 ...

  8. Asp.NET websocket,Asp.NET MVC 使用 SignalR 实现推送功能一(Hubs 在线聊天室)

    ASP .NET SignalR 是一个ASP .NET 下的类库,可以在ASP .NET 的Web项目中实现实时通信.什么是实时通信的Web呢?就是让客户端(Web页面)和服务器端可以互相通知消息及 ...

  9. Asp.Net MVC+EF+三层架构的完整搭建过程

    架构图: 使用的数据库: 一张公司的员工信息表,测试数据 解决方案项目设计: 1.新建一个空白解决方案名称为Company 2.在该解决方案下,新建解决方案文件夹(UI,BLL,DAL,Model) ...

随机推荐

  1. 《JavaScript语言精粹》第二章-语法 简单笔记

    注释 JavaScript提供两种注释: /* */包围的块注释及//开头的行注释. 注释应该被优先用来提高程序的可读性,注释要精确地描述代码,没有用的注释比没有注释更糟糕. /* */块注释对于被注 ...

  2. python写入文件编码报错

    decode的作用是将其他编码的字符串转换成unicode编码,如str1.decode('gb2312'),表示将gb2312编码的字符串str1转换成unicode编码. encode的作用是将u ...

  3. GVIM安装手记

    GVIM安装手记 1. 安装GIT及GVIM Downloa Git URL : https://gitforwindows.org/ Downloa GVim URL : https://www.v ...

  4. burpsuite 抓HTTPS数据包

    抓HTTPS数据包 导出保存为cer证书文件,导入到受信任的根证书颁发机构 设置代理服务器与burp中proxy listeners保持一致 设置目标url 抓包 可用repeater发请求

  5. 2017-10-26 NOIP模拟赛2

    财富 (treasure) Time Limit:1000ms   Memory Limit:128MB 题目描述 LYK有n个小伙伴.每个小伙伴有一个身高hi. 这个游戏是这样的,LYK生活的环境是 ...

  6. 2017-10-5 清北刷题冲刺班a.m

    行列式 序列 #include<iostream> #include<cstdio> #define maxn 500010 using namespace std; int ...

  7. JSONObject,JSONArray,String,Map间的互转

    引言      在平常的Web项目开发过程中,json和String.map是最常用的类型和返回结果集,其中也经常会涉及到之间的各种相互转换,下边就总结一下: 1.String转JSONObject ...

  8. POJ1023 The Fun Number System

    题目来源:http://poj.org/problem?id=1023 题目大意: 有一种有趣的数字系统.类似于我们熟知的二进制,区别是每一位的权重有正有负.(低位至高位编号0->k,第i位的权 ...

  9. Python 简单的方法爬取b站dnf视频封面

    import urllib.request cnt=0 def instr(keystr): st=keystr.find('(')+1 strhtml=keystr[st:len(keystr)-1 ...

  10. Abbreviation ---- hackerrank

    ---恢复内容开始--- https://www.hackerrank.com/contests/world-codesprint-6/challenges/abbr 给定两个串str和sub. 对于 ...