其实网上关于HostingEnvironment 的RegisterObject和QueueBackgroundWorkItem文章已经很多了,典型是的
QueueBackgroundWorkItem to> Fire and Forget on ASP.NET 该文章里面涉及到一个开与项目 AspNetBackgroundTasks
而我本次的测试也是基于AspNetBackgroundTasks来测试的,Fire and Forget on ASP.NET里面提到了3点:个人英文不太好 就把原文贴出来吧
大致意思就是Task.Run, Task.Factory.StartNew, Delegate.BeginInvoke, ThreadPool.QueueUserWorkItem 像这样的code, 在应用程程序域的DomainUnload以后,马上消失;在4.5.2 以后我们可以考虑 HostingEnvironment.QueueBackgroundWorkItem方法,它在 在应用程程序域的DomainUnload以后可以坚持30秒,后面我测试过也差不多就是这个时间。推荐做法就是用 HostingEnvironment.RegisterObject 它坚持的时间更长(我本地测试大致为5分钟)。
先看看 我的demo, 系统中总有一些比较耗时的操作, 通常我们可以采用分布式消息队列来实现,网上发现有人用 BlockingCollection<UserInfo>(new ConcurrentQueue<UserInfo>())来做简单的消息队列,那么我的担心就 出来了,BlockingCollection是线程安全的,但是它毕竟驻留在IIS进程里面,如果在release 跟新的时候,iis程序池会回收,BlockingCollection没有处理的数据是否会丢失?
先看看code吧,AsyncService每隔1秒就处理一条数据
public>{
public string UserName { set; get; } }
public>{public static BlockingCollection<UserInfo> UerQueue;static AsyncService() {
UerQueue
= new BlockingCollection<UserInfo>(new ConcurrentQueue<UserInfo>()); }
public static void Start() {
DateTime start
= DateTime.Now;foreach (UserInfo item in UerQueue.GetConsumingEnumerable()) {
ProcessUserInfo(item,(DateTime.Now
-start).TotalSeconds); Thread.Sleep(
1000); }
}
private static void ProcessUserInfo(UserInfo userInfo,double seconds) {
System.Diagnostics.Debug.WriteLine(userInfo.UserName
+"------------"+seconds.ToString()); }
}
在HomeController往队列里面加数据
public ActionResult Index() {
for (int i = 0; i < 600; i++) {
AsyncService.UerQueue.Add(
new UserInfo { UserName=$"ma jiang -{i}" }); }
return View(); }
我的实际操作是在程序运行在7-10左右就停止IISExpress
在Application_Start方法里面用:
HostingEnvironment.QueueBackgroundWorkItem(x =>
{
AsyncService.Start();
});
运行结果如图:
也就是说我的IISExpress退出后大概运行了30秒。
调用:
BackgroundTaskManager.Run(() =>
{
AsyncService.Start();
});
运行结果如图:
也就是说我的IISExpress退出后大概运行了300秒
这里推荐一个源码地址 HostingEnvironment.cs 通过阅读源码 已经证实Fire and Forget on ASP.NET里面的描述的准确性。虽然我的测试demo不够完善,但是并不影响结论:推荐使用 HostingEnvironment.RegisterObject方法。
demo下载
该demo 在IIS下测试过, 比如:我们复制bin目录下的文件, 修改config文件 只要应用程序池 的进程ID 还存在,该方案一直可行,如
经过测试,用如下code,只要IIS 应用程序池 进程ID不变(或者说 对应的进程ID存在),中间无论修改bin目录下文件,还是配置文件(也就是网上说的什么应用程序池自动回收的那些东东),都不会影响BackgroundTaskManager.Run里面的code。
BackgroundTaskManager.Run(() =>
{
AsyncService.Start();
});
运维网声明
1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网 享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com