Thread.Sleep(0) vs Sleep(1) vs Thread.Yeild()
注:本文为个人学习摘录,原文地址:http://www.cnblogs.com/stg609/p/3857242.html
本文将要提到的线程及其相关内容,均是指 Windows 操作系统中的线程,不涉及其它操作系统。
文章索引
- 核心概念
- Thread.Yeild
- Thread.Sleep(0)
- Thread.Sleep(1)
- 实验告诉你:单一线程
- 实验告诉你:多线程(同优先级)
- 实验告诉你:多线程(不同优先级)
- 本人观点
- 参考资源
在进入正文前,有几个知识点需要大家在阅读前有所了解。
核心概念
优先级调度算法
处理器是一个操作系统执行任务的工具,线程是一个操作系统执行任务的基本单位,处理器的数量决定了不可能所有线程都能同时得到执行。这就需要通过某种算法来进行任务高度。而 Windows 是一个抢占式的多任务操作系统,我们来看下维基百科对于抢占式的定义:
In computing, preemption is the act of temporarily interrupting a task being carried out by a computer system, without requiring its cooperation, and with the intention of resuming the task at a later time. Such a change is known as a context switch. It is normally carried out by a privileged task or part of the system known as a preemptive scheduler, which has the power to preempt, or interrupt, and later resume, other tasks in the system.
上面这段英文意味着在抢占式的操作系统中,执行任务的多个线程之间会因为某种因素互相争抢在处理器上运行的机会。而这种因素就是标题所说的 “优先级”。
线程是根据其优先级而调度执行的。 即使线程正在运行时中执行,所有线程都是由操作系统分配处理器时间片的。 用于确定线程执行顺序的调度算法的详细情况随每个操作系统的不同而不同。
在某些操作系统下,具有最高优先级(相对于可执行线程而言)的线程经过调度后总是首先运行。 如果具有相同优先级的多个线程都可用,则计划程序将遍历处于该优先级的线程,并为每个线程提供一个固定的时间片(段)来执行。 只要具有较高优先级的线程可以运行,具有较低优先级的线程就不会执行。 如果在给定的优先级上不再有可运行的线程,则计划程序将移到下一个较低的优先级并在该优先级上调度线程以执行。 如果此时具有较高优先级的线程可以运行,则具有较低优先级的线程将被抢先,并允许具有较高优先级的线程再次执行。 除此之外,当应用程序的用户界面在前台和后台之间移动时,操作系统还可以动态调整线程优先级。 其他操作系统可以选择使用不同的调度算法。
--- 调度线程
上面文字中提到优先级可以由操作系统动态调整。Windows 除了会在前后台切换的时候调整优先级还会为 I/O 操作动态提升优先级,或者使用 “饥渴” 的时间片分配策略来动态调整,如果有线程一直渴望得到时间片但是很长时间都没有获得时间片,Windows 就会临时将这个线程的优先级提高,并一次分配给2倍的时间片来执行,当用完2倍的时间片后,优先级又会恢复到之前的水平。
线程运行状态
一个线程从开始到终止可能会有上述几种状态,这几种状态可以互相转换。(上图中的 “就绪” 指的是 runnable 状态,又称为 “ready to run” 状态,个人感觉翻译成 “就绪” 比 “可运行” 要来得明白)。
Thread.Yeild
对上述概念有所了解后,我将正式介绍这三个方法的区别。
该方法是在 .Net 4.0 中推出的新方法,它对应的底层方法是 SwitchToThread。
Yield 的中文翻译为 “放弃”,这里意思是主动放弃当前线程的时间片,并让操作系统调度其它就绪态的线程使用一个时间片。但是如果调用 Yield,只是把当前线程放入到就绪队列中,而不是阻塞队列。如果没有找到其它就绪态的线程,则当前线程继续运行。
Yielding is limited to the processor that is executing the calling thread. The operating system will not switch execution to another processor, even if that processor is idle or is running a thread of lower priority. If there are no other threads that are ready to execute on the current processor, the operating system does not yield execution, and this method returns false.
This method is equivalent to using platform invoke to call the native Win32 SwitchToThread function. You should call the Yield method instead of using platform invoke, because platform invoke bypasses any custom threading behavior the host has requested.
优势:比 Thread.Sleep(0) 速度要快,可以让低于当前优先级的线程得以运行。可以通过返回值判断是否成功调度了其它线程。
劣势:只能调度同一个处理器的线程,不能调度其它处理器的线程。当没有其它就绪的线程,会一直占用 CPU 时间片,造成 CPU 100%占用率。
Thread.Sleep(0)
Sleep 的意思是告诉操作系统自己要休息 n 毫秒,这段时间就让给另一个就绪的线程吧。当 n=0 的时候,意思是要放弃自己剩下的时间片,但是仍然是就绪状态,其实意思和 Yield 有点类似。但是 Sleep(0) 只允许那些优先级相等或更高的线程使用当前的CPU,其它线程只能等着挨饿了。如果没有合适的线程,那当前线程会重新使用 CPU 时间片。
If you specify 0 milliseconds, the thread will relinquish the remainder of its time slice but remain ready.
--- Sleep Function
优势:相比 Yield,可以调度任何处理器的线程使用时间片。
劣势:只能调度优先级相等或更高的线程,意味着优先级低的线程很难获得时间片,很可能永远都调用不到。当没有符合条件的线程,会一直占用 CPU 时间片,造成 CPU 100%占用率。
Thread.Sleep(1)
该方法使用 1 作为参数,这会强制当前线程放弃剩下的时间片,并休息 1 毫秒(因为不是实时操作系统,时间无法保证精确,一般可能会滞后几毫秒或一个时间片)。但因此的好处是,所有其它就绪状态的线程都有机会竞争时间片,而不用在乎优先级。
优势:可以调度任何处理器的线程使用时间片。无论有没有符合的线程,都会放弃 CPU 时间,因此 CPU 占用率较低。
劣势:相比 Thread.Sleep(0),因为至少会休息一定时间,所以速度要更慢。
实验告诉你:单一线程
测试环境:Windows 7 32位、VMWare workstation、单处理器单核芯
开发环境:Visual Studio 2012、控制台项目、Release 编译后将 exe 拷贝到虚拟机后运行
Thread.Yeild
static void Main()
{
string s = "";
while (true)
{
s = DateTime.Now.ToString(); //模拟执行某个操作
Thread.Yeild();
}
}
执行效果
Thread.Sleep(0)
static void Main()
{
string s = "";
while (true)
{
s = DateTime.Now.ToString();
Thread.Sleep();
}
}
执行效果
Thread.Sleep(1)
static void Main()
{
string s = "";
while (true)
{
s = DateTime.Now.ToString();
Thread.Sleep();
}
}
执行效果
通过上述三个实验,很明显说明 Thread.Sleep(1) 对于解决资源 100% 占用是有明显效果的。
实验告诉你:多线程(同优先级)
我的实验方法很简单,就是通过 while 让该线程不断的执行,为了让大家一目了然两个线程的交替,通过向控制台输出不同的字符串来验证。
while 语句执行速度相当快,所以必须所以加上一些代码来浪费CPU时间,以免大家只能看到不断的刷屏。
辅助代码
private static void WasteTime()
{
// 耗时约 200ms
DateTime dt = DateTime.Now;
string s = "";
while (DateTime.Now.Subtract(dt).Milliseconds <= )
{
s = DateTime.Now.ToString(); //加上这句,防止被编译器优化
}
}
不使用任何方法
让线程们自己争用 CPU 时间。
static void Main(string[] args)
{
Thread t = new Thread(() =>
{
while (true)
{
WasteTime();
Console.WriteLine("Thread 1 ==========");
}
});
t.IsBackground = true;
Thread t2 = new Thread(() =>
{
while (true)
{
WasteTime();
Console.WriteLine("Thread 2");
}
}); t2.IsBackground = true;
t2.Start();
t.Start();
Console.ReadKey(); t.Abort();
t2.Abort();
Console.ReadKey();
}
执行效果
Thread.Yeild
修改第一个线程的方法体,加入Thread.Yield。其余代码不变。
Thread t = new Thread(() =>
{
while (true)
{
WasteTime();
Thread.Yield();
Console.WriteLine("Thread 1 ==========");
}
});
执行效果
Thread.Sleep(0)
仿照 Thread.Yield,只不过用 Thread.Sleep(0)替换。
执行效果
Thread.Sleep(1)
仿照 Thread.Yield,用 Thread.Sleep(1)替换。
执行效果
从上面的示例看出,未使用 Thread 的这几个方法(Yield,Sleep)的例子,两个同等优先级的线程可以获得差不多完全一样的CPU时间。
而使用了 Thread 方法的那几个例子或多或少都让另一个线程多获取了些时间片,但是不同的方法执行效果差得并不多。这是因为它们所让出的时间片往往都只是几十毫秒的事情,这么短的时间,对于我的测试代码来说很难顺利扑捉每个瞬间。
Thread2 要得到更多的时间片
实验告诉你:多线程(不同优先级)
先来调整下代码,修改 Thread 1,让它的优先级变成 “AboveNormal”。
Thread t = new Thread(() =>
{
while (true)
{
WasteTime();
Console.WriteLine("Thread 1 ==========");
}
}); t.Priority = ThreadPriority.AboveNormal; // 加入这句话
t.IsBackground = true;
不使用任何方法
Thread t = new Thread(() =>
{
while (true)
{
WasteTime();
Console.WriteLine("Thread 1 ==========");
}
}); t.Priority = ThreadPriority.AboveNormal;
t.IsBackground = true; Thread t2 = new Thread(() =>
{
while (true)
{
WasteTime();
Console.WriteLine("Thread 2");
} }); t2.IsBackground = true;
执行效果
从实验中可以表明,低优先级的几乎很少有使用时间片的时候。
Thread.Yeild
修改 Thread 的方法体
Thread t = new Thread(() =>
{
while (true)
{
WasteTime();
Console.WriteLine("Thread 1 ========== {0}",Thread.Yield());
}
});
执行效果
使用了 Yeild 之后,很明显低优先级的线程现在也能够获得CPU时间了。
Thread.Sleep(0)
修改方法体
Thread t = new Thread(() =>
{
while (true)
{
WasteTime();
Console.WriteLine("Thread 1 ==========");
Thread.Sleep();
}
});
执行效果
有没有发现低优先级的线程又一次被无情的抛弃了?
Thread.Sleep(1)
仿照 Sleep(0),用 Sleep(1)替换。
执行效果
通过上面的实验,我想你应该已经比较清楚了在不同优先级的情况下,哪个方法更适用于去切换线程。
本人观点
有鉴于 Thread.Sleep(0) 的表现,本人认为应该无情的把它取缔掉。至于是用 Thread.Yeild,还是 Thread.Sleep(n) (n>0),那就根据实际情况吧。欢迎大家补充~
参考资源
Consequences of the scheduling algorithm: Sleeping doesn't always help
SwitchToThread/Thread.Yield vs. Thread.Sleep(0) vs. Thead.Sleep(1)
本文来源《C# 基础回顾: Thread.Sleep(0) vs Sleep(1) vs Yeild》
Thread.Sleep(0) vs Sleep(1) vs Thread.Yeild()的更多相关文章
- Thread.Sleep(0) vs Sleep(1) vs Yeild
本文将要提到的线程及其相关内容,均是指 Windows 操作系统中的线程,不涉及其它操作系统. 文章索引 核心概念 Thread.Yeild Thread.Sleep(0) Thread. ...
- 【转】Thread.sleep(0)的意义
Thread.sleep(0)的意义 2012-03-23 17:47 2188人阅读 评论(2) 收藏 举报 windows算法unixthread 我们可能经常会用到 Thread.Sleep 函 ...
- Thread系列之Thread.Sleep(0)
线程这一概念,可以理解成进程中的一个小单元.这个单元是一个独立的执行单元,但是与进程中的其他线程共享进程中的内存单元. 由于Cpu资源是有限的,所以进程中的多个线程要抢占Cpu,这也导致进程中的多个线 ...
- Thread.sleep(0)的意义& 多线程
我们可能经常会用到 Thread.Sleep 函数来使线程挂起一段时间.那么你有没有正确的理解这个函数的用法呢?思考下面这两个问题: 假设现在是 2008-4-7 12:00:00.000,如果我调用 ...
- 关于Thread.Sleep(0)
看到的文章,写的不错. 我们可能经常会用到 Thread.Sleep 函数来使线程挂起一段时间.那么你有没有正确的理解这个函数的用法呢?思考下面这两个问题:假设现在是 2008-4-7 12:00:0 ...
- 对线程调度中Thread.sleep(0)的深入理解
在Java或者C#中,都会用到 Thread.Sleep()来使线程挂起一段时间.那么你有没有正确的理解这个方法的用法呢?思考下面这两个问题: 1.假设现在是 2014-8-13 17:00:00.0 ...
- Thread系列——Thread.Sleep(0)
转载自:http://www.cnblogs.com/ATually/archive/2010/10/21/1857261.html 线程这一概念,可以理解成进程中的一个小单元.这个单元是一个独立的执 ...
- 说说Thread.Sleep(0)的那些奇怪的事
写在前面 最近在弄一个传输组件,用到很多多线程的知识,其中有个问题,困扰我很久,不知道是什么原因,脑子一热,在传输过程中,添加了一句代码Thread.Sleep(0).那个问题竟然解决了,耗费我一上午 ...
- [转载]Thread.Sleep(0)妙用
原文地址:http://blog.csdn.net/lgstudyvc/article/details/9337063 来自本论坛: 我们可能经常会用到 Thread.Sleep 函数来使线程挂起一段 ...
随机推荐
- 真与假与c#,java中的不同之处
/************真与假************/ /*C语言中:真(非0).假(0) * Java.C#中:真(true).假(false) * JavaScript中:真(非0.true. ...
- Web前端开发的一点记录
工欲善其事必先利其器,开发工具选择Sublime Text 简称(ST) 本文所说的均在Windows NT 环境下使用的ST3运行. ST的Package Control安装方法: 1. 直接输入p ...
- 自定义HttpFilter模块完善
自定义HttpFilter模块完善 背景 在12月由于要针对项目做用户操作日志,但不想在每个方法里去增加代码,写入用户日志.因为这样具体的方法违背职责单一的原则,若后期日志内容格式发生变更,或其他 ...
- Binder机制,从Java到C (5. IBinder对象传递形式)
1.IBinder的传递 Binder IPC通信中,Binder是通信的媒介,Parcel是通信的內容.远程调用过程中,其参数都被打包成Parcel的形式來传递.IBinder对象当然也不例外,在前 ...
- ubuntu下无法打开windows下ntfs文件系统的解决方法
例如:/dev/sda5无法加载,使用如下命令 sudo ntfsfix /dev/sda5 如果命令没有安装,先安装(sudo apt-get install ntfsprogs).
- go语言defer使用
defer Go语言中有种不错的设计,即延迟(defer)语句,你可以在函数中添加多个defer语句.当函数执行到最后时,这些defer语句会按照逆序执行,最后该函数返回.特别是当你在进行一些打开资源 ...
- iOS关于RunLoop和Timer
RunLoop这个东西,其实我们一直在用,但一直没有很好地理解它,或者甚至没有知道它的存在.RunLoop可以说是每个线程都有的一个对象,是用来接受事件和分配任务的loop.永远不要手动创建一个run ...
- 基于Redis的CustomerSessionProvider(一)
CustomerSessionProvider需要实现SessionStateStoreProviderBase 在设计Redis Session Provider的时候,需要考虑 1.是否每个用户的 ...
- 服务端套接字类CxServerSocket的使用
服务端套接字类CxServerSocket的使用 这是一个精练的服务端套接字类,类名.函数名和变量名均采用匈牙利命名法.小写的x代表我的姓氏首字母(谢欣能),个人习惯而已,如有雷同,纯属巧合. CxS ...
- js的onclick和jquery的bind事件执行先后顺序
近期在项目中为每一个ajax触发按钮写正在加载的效果,用的是bootstarp 代码如下 $(function(){ $('.btn').bind('click',function(e){ var $ ...