C# System.Timers.Timer中的坑,程序异常退出后timer依然运行问题
问题背景
C#小白,由于本公司IM系统服务端(java)是本人独立开发的,加上现在所在项目需要对接IM系统,于是IM的客户端(C#实现)对接工作就交给我了。于是C#小白的我天真的以为只要调用C#端的SDK接口真搞定了。起初都还好,对接工作都很正常,没什么大问题。可是随着时间的不断流逝,终于在项目组小伙伴的不断使用中发现经常登不上IM系统,然而让我过去调试的时候又发现是正常的,让人很抓狂有木有!
直到他们再次重现BUG给我看的时候,我发现服务端日志中不断收到登录请求,而且他们程序关闭之后还是这样,小伙伴们百思不得其解,都怀疑我的服务端代码有问题。对服务端代码很熟悉的我自然不会认为服务端会有什么问题,因为登录只有收到客户端的UDP请求才会触发,而客户端不断发送登录请求这种情况也只有客户端自动重连机制了。而客户端正常退出时 根本不会触发自动重连机制,因为正常退出时会发出登出请求,并且会释放自动重连,心跳以及QoS等资源,那就只有异常退出时可能会出现这种问题了,如:任务管理器强制杀死程序、程序卡死、闪退等情况,他们在开发调试的时候也经常通过IDE直接关闭。于是我做了一个实验,在客户端登录成功后,反复n次正常登陆登出,没有任何问题;然后又尝试了异常退出,虽然没有立即登出IM系统(没发登出请求嘛),但也不会出现疯狂重连的情况,当然10s没有心跳服务器自然认为用户掉线了,一切正常;下面开始重头戏了,我改了客户端SDK的代码,在登陆成功后直接调用自动重连机制,然后任务管理器强制杀死程序,果不其然,BUG重现了。。。
问题探究
问题已经很明确了,非正常情况下的程序关闭并且自动重连程序在执行中的时候服务端会不断收到登录请求。到底是什么原因导致的呢?于是我不(忐)慌(忑)不忙(安)的去看了下客户端SDK中自动重连部分的代码,这里主要依靠timer定时器每隔2s发一次登录请求,直到登录成功为止。琢磨了一会儿也没发现有什么问题,慢慢的我开始猜测会不会是这timer有问题?于是把timer换成了Thread,通过Thread.sleep(2000)加是否重连成功的状态位来达到定时的效果,然后心怀期待的去测试,依然是在登陆成功后直接调用自动重连机制,然后任务管理器强制杀死程序,BUG消失了。一时兴奋的我多测试了几次,终于可以确定是timer的问题了,欣喜若狂的我心里不免开始了咒骂,哪有进程都杀死了,线程还在跑的道理嘛!!!
解决问题
C#中的timer定时器分为三种:基于服务器的计时器System.Timers.Timer、线程计时器System.Threading.Timer以及基于 Windows 的标准计时器System.Windows.Forms.Timer。其中第一种和第二种都是通过threadpool新开一个线程,因此效率较高;而第三种则是通过windows消息机制实现,和它所属的Form属于同一个线程,故效率较低。由于IM是不依赖于窗体的,属于后台独立线程,故排除第三种的使用可能。而IMSDK中使用的是第一种System.Timers.Timer而出现的问题,故我选择使用第二种线程计时器System.Threading.Timer来代替,经过大量测试后,完美解决问题,故记录一番。
至于System.Timers.Timer为什么会出现这种情况,在下学疏才浅,实在不知道为什么,希望知道的大神能点拨一二,不胜感激!
C# System.Timers.Timer中的坑,程序异常退出后timer依然运行问题的更多相关文章
- 使用System.Timers.Timer类实现程序定时执行
使用System.Timers.Timer类实现程序定时执行 在C#里关于定时器类有3个:System.Windows.Forms.Timer类.System.Threading.Timer类和Sys ...
- .NET中System.Diagnostics.Stopwatch、System.Timers.Timer、System.Threading.Timer 的区别
1.System.Diagnostics.Stopwatch Stopwatch 实例可以测量一个时间间隔的运行时间,也可以测量多个时间间隔的总运行时间. 在典型的 Stopwatch 方案中,先调用 ...
- .NET System.Timers.Timer的原理和使用(开发定时执行程序)
概述(来自MSDN) Timer 组件是基于服务器的计时器,它使您能够指定在应用程序中引发Elapsed 事件的周期性间隔.然后可以操控此事件以提供定期处理.例如,假设您有一台关键性服务器,必须每周7 ...
- 简述System.Windows.Forms.Timer 与System.Timers.Timer用法区别
System.Windows.Forms.Timer 基于窗体应用程序 阻塞同步 单线程 timer中处理时间较长则导致定时误差极大. System.Timers.Timer 基于服务 非阻塞异步 多 ...
- System.Timers.Timer
前言 System.Timers.Timer组件是基于服务器的计时器,它能够指定在应用程序中引发Elapsed事件周期性间隔,以处理相应事件. 使用示例: 运行结果展示: System.Timers. ...
- C# System.Timers.Timer的一些小问题?
比如设置间隔时间是 1000msSystem.Timers.Timer mytimer = new System.Timers.Timer(1000);问题若响应函数执行的时间超过了 1000 ms, ...
- [C#]System.Timers.Timer
摘要 在.Net中有几种定时器,最喜欢用的是System.Timers命名空间下的定时器,使用起来比较简单,作为定时任务,有Quartz.net,但有时候,一个非常简单的任务,不想引入这个定时任务框架 ...
- System.Windows.Forms.Timer与System.Timers.Timer的区别(zz)
.NET Framework里面提供了三种Timer: System.Windows.Forms.Timer System.Timers.Timer System.Threading.Timer VS ...
- .Net Windows Service(服务) 调试安装及System.Timers.Timer 使用
Windows Service(服务) 是运行在后台的进程 1.VS建立 Windows 服务(.NET Framework) 2.添加Timer 双击Service1.cs可以拖控件(System ...
随机推荐
- CSS 的overflow:hidden (清除浮动)
verflow:hidden这个CSS样式是大家常用到的CSS样式,但是大多数人对这个样式的理解仅仅局限于隐藏溢出,而对于清除浮动这个含义不是很 了解.一提到清除浮动,我们就会想到另外一个CSS样式: ...
- Cat- Linux必学的60个命令
1.作用 cat(“concatenate”的缩写)命令用于连接并显示指定的一个和多个文件的有关信息,它的使用权限是所有用户. 2.格式 cat [options] 文件1 文件2…… 3.[opti ...
- LUOGU P4074 [WC2013]糖果公园 (树上带修莫队)
传送门 解题思路 树上带修莫队,搞了两天..终于开O2+卡常大法贴边过了...bzoj上跑了183s..其实就是把树上莫队和带修莫队结合到一起,首先求出括号序,就是进一次出一次那种的,然后如果求两个点 ...
- 关于HTTP协议(转)
HTTP简介 HTTP协议是Hyper Text Transfer Protocol(超文本传输协议)的缩写,是用于从万维网(WWW:World Wide Web )服务器传输超文本到本地浏览器的传送 ...
- win 7下安装mysql zip格式
mysql 下载地址:https://dev.mysql.com/downloads/mysql/5.5.html#downloads 下载的mysql格式为zip: 下载完成放在除C盘以外的盘. 一 ...
- org.hibernate.service.spi.ServiceException: Unable to create requested service [org.hibernate.engine.jdbc.connections.spi.ConnectionProvider]
从hibernate3升级到4应该会遇到 应该添加引用 <!-- https://mvnrepository.com/artifact/org.hibernate/hibernate-c3p0 ...
- 简单易学的机器学习算法——基于密度的聚类算法DBSCAN
一.基于密度的聚类算法的概述 最近在Science上的一篇基于密度的聚类算法<Clustering by fast search and find of density peaks> ...
- css3之弹性盒模型(Flex Box)
CSS3 弹性盒子(Flex Box) 弹性盒子是 CSS3 的一种新的布局模式. CSS3 弹性盒( Flexible Box 或 flexbox),是一种当页面需要适应不同的屏幕大小以及设备类型时 ...
- JSP-案例-商品增删改
商品的增删改查 1显示 部分代码 Dao public List<Product> findAllProduct() throws SQLException { QueryRunner r ...
- Leetcode89. Gray Code格雷编码
给定一个代表编码总位数的非负整数 n,打印其格雷编码序列.格雷编码序列必须以 0 开头. 示例 1: 输入: 2 输出: [0,1,3,2] 解释: 00 - 0 01 - 1 11 - 3 10 - ...