:本文为个人学习摘录,原文地址:http://www.cnblogs.com/stg609/p/3857242.html

  本文将要提到的线程及其相关内容,均是指 Windows 操作系统中的线程,不涉及其它操作系统。

  文章索引

  在进入正文前,有几个知识点需要大家在阅读前有所了解。

核心概念

  优先级调度算法

  处理器是一个操作系统执行任务的工具,线程是一个操作系统执行任务的基本单位,处理器的数量决定了不可能所有线程都能同时得到执行。这就需要通过某种算法来进行任务高度。而 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.

--- Preemption (computing)

  上面这段英文意味着在抢占式的操作系统中,执行任务的多个线程之间会因为某种因素互相争抢在处理器上运行的机会。而这种因素就是标题所说的 “优先级”。

线程是根据其优先级而调度执行的。 即使线程正在运行时中执行,所有线程都是由操作系统分配处理器时间片的。 用于确定线程执行顺序的调度算法的详细情况随每个操作系统的不同而不同。

在某些操作系统下,具有最高优先级(相对于可执行线程而言)的线程经过调度后总是首先运行。 如果具有相同优先级的多个线程都可用,则计划程序将遍历处于该优先级的线程,并为每个线程提供一个固定的时间片(段)来执行。 只要具有较高优先级的线程可以运行,具有较低优先级的线程就不会执行。 如果在给定的优先级上不再有可运行的线程,则计划程序将移到下一个较低的优先级并在该优先级上调度线程以执行。 如果此时具有较高优先级的线程可以运行,则具有较低优先级的线程将被抢先,并允许具有较高优先级的线程再次执行。 除此之外,当应用程序的用户界面在前台和后台之间移动时,操作系统还可以动态调整线程优先级。 其他操作系统可以选择使用不同的调度算法。

--- 调度线程

  上面文字中提到优先级可以由操作系统动态调整。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.Yield Method

  优势:比 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()的更多相关文章

  1. Thread.Sleep(0) vs Sleep(1) vs Yeild

    本文将要提到的线程及其相关内容,均是指 Windows 操作系统中的线程,不涉及其它操作系统. 文章索引 核心概念 Thread.Yeild       Thread.Sleep(0) Thread. ...

  2. 【转】Thread.sleep(0)的意义

    Thread.sleep(0)的意义 2012-03-23 17:47 2188人阅读 评论(2) 收藏 举报 windows算法unixthread 我们可能经常会用到 Thread.Sleep 函 ...

  3. Thread系列之Thread.Sleep(0)

    线程这一概念,可以理解成进程中的一个小单元.这个单元是一个独立的执行单元,但是与进程中的其他线程共享进程中的内存单元. 由于Cpu资源是有限的,所以进程中的多个线程要抢占Cpu,这也导致进程中的多个线 ...

  4. Thread.sleep(0)的意义& 多线程

    我们可能经常会用到 Thread.Sleep 函数来使线程挂起一段时间.那么你有没有正确的理解这个函数的用法呢?思考下面这两个问题: 假设现在是 2008-4-7 12:00:00.000,如果我调用 ...

  5. 关于Thread.Sleep(0)

    看到的文章,写的不错. 我们可能经常会用到 Thread.Sleep 函数来使线程挂起一段时间.那么你有没有正确的理解这个函数的用法呢?思考下面这两个问题:假设现在是 2008-4-7 12:00:0 ...

  6. 对线程调度中Thread.sleep(0)的深入理解

    在Java或者C#中,都会用到 Thread.Sleep()来使线程挂起一段时间.那么你有没有正确的理解这个方法的用法呢?思考下面这两个问题: 1.假设现在是 2014-8-13 17:00:00.0 ...

  7. Thread系列——Thread.Sleep(0)

    转载自:http://www.cnblogs.com/ATually/archive/2010/10/21/1857261.html 线程这一概念,可以理解成进程中的一个小单元.这个单元是一个独立的执 ...

  8. 说说Thread.Sleep(0)的那些奇怪的事

    写在前面 最近在弄一个传输组件,用到很多多线程的知识,其中有个问题,困扰我很久,不知道是什么原因,脑子一热,在传输过程中,添加了一句代码Thread.Sleep(0).那个问题竟然解决了,耗费我一上午 ...

  9. [转载]Thread.Sleep(0)妙用

    原文地址:http://blog.csdn.net/lgstudyvc/article/details/9337063 来自本论坛: 我们可能经常会用到 Thread.Sleep 函数来使线程挂起一段 ...

随机推荐

  1. ASP.NET MVC应用程序使用axd格式文件

    ASP.NET MVC应用程序使用axd格式文件 axd格式文件,不管是在asp.net还是现在开发asp.net MVC应用程序,都是Insus.NET较喜欢使用的. 因为我们可以虚拟一个在应用程序 ...

  2. boost------ref的使用(Boost程序库完全开发指南)读书笔记

    STL和Boost中的算法和函数大量使用了函数对象作为判断式或谓词参数,而这些参数都是传值语义,算法或函数在内部保修函数对象的拷贝并使用,例如: #include "stdafx.h&quo ...

  3. JS实现等比例缩放图片

    JS实现等比例缩放图片 2014-01-19 21:57 by 龙恩0707, 40 阅读, 0 评论, 收藏, 编辑 JS实现等比例缩放图片 有时候我们前端页面只有500×500像素的宽和高的布局, ...

  4. iOS基础 - iOS程序启动原理

    一.UIApplicationMain 在main.m的main函数中执行了UIApplicationMain这个方法,这是ios程序的入口点 int UIApplicationMain(int ar ...

  5. ETHREAD APC

    ETHREAD APC <寒江独钓>内核学习笔记(4) 继续学习windows 中和线程有关系的数据结构: ETHREAD.KTHREAD.TEB 1. 相关阅读材料 <window ...

  6. EPROCESS KPROCESS PEB

    EPROCESS KPROCESS PEB <寒江独钓>内核学习笔记(2)     在学习笔记(1)中,我们学习了IRP的数据结构的相关知识,接下来我们继续来学习内核中很重要的另一批数据结 ...

  7. 【译】Objectively Speaking 2: A Crash Course in Objective-C for iOS 6

    In this Objective-C tutorial, you will create a simple movie quotes quiz app. Along the way, you’ll ...

  8. logstash+elasticsearch+kibana快速搭建日志平台

    使用logstash+elasticsearch+kibana快速搭建日志平台   日志的分析和监控在系统开发中占非常重要的地位,系统越复杂,日志的分析和监控就越重要,常见的需求有: 根据关键字查询日 ...

  9. 网络负载均衡环境下wsHttpBinding+Message Security+Windows Authentication的常见异常

    提高Windows Communication Foundation (WCF) 应用程序负载能力的方法之一就是通过把它们部署到负载均衡的服务器场中. 其中可以使用标准的负载均衡技术, Windows ...

  10. OpenCascade

    Hello World of OpenCascade   Hello World of OpenCascade eryar@163.com 摘要Abstract:以一个经典的Hello World程序 ...