为什么要放弃使用Thread.Sleep
前言
此文并不是说要完全放弃使用Thread.Sleep,而是要说明在符合哪些情况下使用!
场景
很多时候,我们会需要一个定时服务来处理业务。
但并不是死死的每隔N分钟执行一次那种,而是在一次处理完后,算好下一次处理的时间点。
当到达此时间点,触发程序重新开始执行代码。
普遍做法
普遍的情况下,都是使用while(true){Thread.Sleep()}来实现,废话不多话,看代码版本1:
class Program
{
static void Main(string[] args)
{
var workLists = new List<string>() { "任务1", "任务2", "任务3", "任务4" };
foreach (var task in workLists)
{
var thread = new System.Threading.Thread(new System.Threading.ParameterizedThreadStart(Work.DoWork));
thread.Start(task);
}
}
}
class Work
{
public static void DoWork(object target)
{
var taskType = target as string;
var interval = 1 * 60 * 1000;//处理失败,1分钟后重试
var maxTimes = 5;
var retryTimes = 0;
while (true)
{
while (retryTimes < maxTimes)
{
var ok = Proccess(taskType);
if (ok)
{
retryTimes = maxTimes;
}
else
{
retryTimes++;
System.Threading.Thread.Sleep(interval);
}
}
var tim = GetTotalMillisecondsForNext();//计算离下一次开始处理的时间
System.Threading.Thread.Sleep(tim);//挂起一段时间后,重新唤醒
retryTimes = 0;
}
}
private static bool Proccess(string taskType)
{
Console.WriteLine("开始执行处理:{0}", taskType);
return true;
}
private static int GetTotalMillisecondsForNext()
{
//这里根据自己的业务来决定
return 2 * 1000;
}
}
代码简单易懂。
分析
版本1中,循环强制创建线程,并使用System.Threading.Thread.Sleep(tim)来挂起线程,然后重新唤醒。
这种方式不好之处在于:占用系统线程资源,是一种浪费。如同占着茅坑不拉屎!线程是一种十分宝贵的资源,创建,销毁,切换 都是相当耗性能的。
当Sleep的时候,就等于说:现在我不用,但是你也别想用。你要用?自己去Create一个。
有的人说,Sleep的时候 不占用CPU啊!对,是不占用CPU ,但是占着线程资源,阻碍系统的线程调度!
可以参考下这文章
Threads are a limited resource, they take approximately 200,000 cycles to create and about 100,000 cycles to destroy. By default they reserve 1 megabyte of virtual memory for its stack and use 2,000-8,000 cycles for each context switch. This makes any waiting thread a huge waste.
改进
使用System.Timers.Timer来改进我们的程序。当执行处理业务的代码时,首先把timer停止,处理完毕后,算好一次执行的时间点,赋给timer并启动,看代码版本2
class Program
{
static void Main(string[] args)
{
var workLists = new List<string>() { "任务1", "任务2", "任务3", "任务4" };
Parallel.ForEach(workLists,
},
(task) => { new Work2() { TaskType = task }.DoWork(); });
Console.ReadLine();
}
}
class Work2
{
private Timer _workTimer;
public string TaskType { get; set; }
public void DoWork()
{
_workTimer = new System.Timers.Timer();
_workTimer.Interval = ;
_workTimer.Elapsed += new ElapsedEventHandler(TimerHanlder);
_workTimer.Start();
}
private void TimerHanlder(object sender, ElapsedEventArgs e)
{
_workTimer.Stop();
* * ;//处理失败,1分钟后重试
;
;
while (retryTimes < maxTimes)
{
var ok = Proccess();
if (ok)
{
retryTimes = maxTimes;
}
else
{
retryTimes++;
System.Threading.Thread.Sleep(interval);
}
}
var times = GetTotalSecondsForNext();
Console.WriteLine("{0}秒后重新执行", times);
_workTimer.Interval = times * ;//计算离下一次开始处理的时间
_workTimer.Start();
}
private bool Proccess()
{
Console.WriteLine("开始执行处理:{0}", TaskType);
return true;
}
private int GetTotalSecondsForNext()
{
//这里根据自己的业务来决定
;
}
}
特别说明一下:Main方法中的Console.ReadLine();很重要,让主线程处于等待的状态,子线程就可以一直执行下去不中断
总结
1:使用Task,而不是使用new System.Threading.Thread。是否要创建线程,应该让系统来决定,利用可复用资源
2: System.Threading.Thread.Sleep(interval);只合适在 "有限度的 " 循环场景中,比如 最多重试N次、倒计时等等
如果不对之处,请各位斧正!
为什么要放弃使用Thread.Sleep的更多相关文章
- 为什么要放弃 JSP?他们终于给出了答案
前言 以前的项目大多数都是Java程序猿又当爹又当妈,既搞前,又搞后端. 随着时代的发展,渐渐的许多大中小公司开始把前后端的界限分的越来越明确,前端工程师只负责前端的事情,后端工程师只管后端的事情.正 ...
- it小小鸟
It小小鸟观后感 每个人的理想目标都是不同的,很多人有自己的理想.却被困于现实而止步不前.一篇<it小小鸟>让我却懂得,一个人如果想有所作为,就不能止步不前.光有一个远大的理想是然并卵的. ...
- The Zero
TOday is just a alpha, but there isnot 欧米伽. 编程,是你选的,是你学的,也是你喜欢的,更是你追求的.为什么要放弃塔!. 加油! 为自己. #include&l ...
- Ueditor上传图片到本地改造到上传图片到云存储
作为新手说多了都是泪啊!我特别想记录一下作为菜鸟时的坑.看看以后是否会看着笑出来. 为什么要改到云存储上就不说了.好处多多. 视频教程上使用的又拍云同时也提到了七牛云.下来我自己也查了下.又拍云是试用 ...
- 从底层角度看ASP.NET-A low-level Look at the ASP.NET...
从更低的角度 这篇文章在一个底层的角度来关注一个web请求怎样到达asp.net框架,从web服务器,通过ISAPI.看看这些后面发生了什么,让我们停止对asp.net的黑箱猜想.ASP.NET是一个 ...
- 所有设计复杂的ORM都是浮云
很久没有写文章了. 一直很忙,不是很有时间整理. 今天主要是来吐槽下那些设计很复杂的ORM的. 项目做的越多,越觉得ORM这个东西设计的太复杂实在是没什么意义. 比较推崇Dapper这样比较简单,效率 ...
- THUWC2019爆零记
Day -1 现在在机房里,准备敲敲板子什么的. 今天晚上放假诶,要好好睡一下.好好睡是不可能的,这辈子不可能的. Day 0 现在在酒店,\(lwh\)神仙在超越,我打了个\(treap\)的板子就 ...
- Appium+python自动化5-Appium Inspector
前言 appium Inspector从入门到放弃!反正你都打开了,那就看下为什么要放弃吧! Appium Inspector是appium自带的一个元素定位工具,上一篇介绍了如何使用uiaut ...
- 关于ADO一个容易被忽视的问题!UpdateBatch [问题点数:0分]
这是一个常见但容易被忽视的问题,旧贴有问及但没答案,因此提高分数.相信大家常这样使用Cache模式:ADOConnection1.BeginTrans;Try ADODataSet1.UpdateBa ...
随机推荐
- 关于本地缓存localStorage
localStorage的优势 1.localStorage拓展了cookie的4K限制 2.localStorage会可以将第一次请求的数据直接存储到本地,这个相当于一个5M大小的针对于前端页面的数 ...
- JAVA可移植性广泛应用
一.JAVA作为一种编程语言:源代码可移植性 作为一种编程语言,JAVA提供了一种最简单同时也是人们最熟悉的可移植性–源代码移植.这意味着任意一个JAVA程序,不论它运行在何种CPU.操作系统或JAV ...
- iOS判断数组不为空
用([array count]==0 )来判断是否为空,都是坑,如果array为空的话,执行count就会直接报错,程序崩溃退出. 正确判断NSArray是否为空的方法: if(array != ni ...
- Process类
public bool Process.Start () 启动(或重用)此 Process 组件的 StartInfo 属性指定的进程资源,并将其与该组件关联. public bool Process ...
- 基本套接字编程(7) -- udp篇
1. UDP概述 UDP 是User Datagram Protocol的简称, 中文名是用户数据报协议,是OSI(Open System Interconnection,开放式系统互 ...
- C#中的BackgroundWorker控件+Delegate.Invoke (委托同步调用)
C#中的BackgroundWorker控件+Delegate.Invoke (委托同步调用) 简单代码,记录一下.一个BackgroundWorker控件 backgroundWorkerRefr ...
- Linux最常用命令之cd和ls
为什么说是最常用的命令呢,因为从普及程度看,即使不怎么接触过Linux系统的人,大多数都会知道这两个命令:而从使用频率看,这两个命令也是当之无愧的首位.现在我们就来看看这两个命令. cd 篇:cd 即 ...
- VS2015调试UWP程序时提示错误DEP0700 : Registration of the app failed. Another user has already installed
在同一台windows10电脑上调试过一个工程以后,切换了账号再次调试出现错误 DEP0700 : Registration of the app failed. Another user has a ...
- shell 条件判断语句整理
常用系统变量 1) $0 当前程式的名称 2) $n 当前程式的第n个参数,n=1,2,…9 3) $* 当前程式的任何参数(不包括程式本身) 4) ...
- Asp.Net异步编程-使用了异步,性能就提升了吗?
Asp.Net异步编程 写在前面的话,很久没有写Blog了,不对,其实一致就没有怎么写过.今天有空,我也来写一篇Blog 随着.Net4.5的推出,一种新的编程方式简化了异步编程,在网上时不时的也看到 ...