C#多线程面面观

转载自:C#多线程面面观

前段时间总算过了把面试别人的瘾(虽然只参与了几个电话面试),经过几个星期总算尘埃落定,新人也已报道。项目组当时准备了一些题,用来考察他们的基本功。总的来说大伙都能说出点,唯一空白的是多线程,无一例外的都说没接触过多线程,也不了解多线程。我就感觉很奇怪,两年左右的程序员竟然直接说不了解多线程(我感觉就算是为了换工作也应该突击了解下的,毕竟我们的招聘要求上也是有了解多线程这条的)。为此我决定总结一下平时用到的多线程知识,写博客前特地又把CLR的线程部分看了一遍,确实每一次看都会有所收获,当然了写例子时也参考了msdn。本文主要以应用为主,原理性的知识就不细讲了。

        主要分为八个部分:1.线程Thread;2.后台线程BackGroundWorker;3.互斥锁Monitor、Lock、Mutex;4.线程池ThreadPool;5.线程间通信WaitHandle、ManualResetEvent、AutoResetEvent和Barrier;6.任务Task、TaskFactory和Parallel;7.异步委托;8.其他,UI多线程,Timer等;

        1.线程Thread

        首先最常用的莫过于Thread了,运用Thread.Sleep(1000);就可以让当前线程休眠1000毫秒;创建一个线程Thread thread=new Thread(action);其中action可以是一个委托或方法名,然后thread.Start();就可以了;默认情况下创建的线程属于前台线程;代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     public class Thread0810  
  7.     {  
  8.         public static void Run()  
  9.         {  
  10.             CheckBackground();  
  11.             Console.ReadLine();  
  12.         }  
  13.   
  14.         private static void CheckBackground()  
  15.         {  
  16.             bool isBackground = true;  
  17.             Thread thread = new Thread(() =>   
  18.             {   
  19.                 Console.WriteLine("Hello Thread!");   
  20.             });  
  21.             isBackground = thread.IsBackground;  
  22.             thread.Start();  
  23.             Thread.Sleep(1 * 1000);  
  24.             Console.WriteLine("IsBackground:" + isBackground.ToString());  
  25.         }  
  26.     }  
  27. }  

        2.后台线程BackGroundWorker

        其次就是后台线程BackGroundWorker了,顾名思义后台线程默认的线程是后台的;前台线程和后台线程的区别在于后台线程在所有的前台线程都结束时会被强制结束;可以注册DoWork事件(注意当注册多个时,事件是一个个的顺序执行,事件绑定多个方法时就是这么执行的),可以设置后台线程是否可以取消;可以设置报告后台线程的进度(需要在执行的方法中调用ReportProgress方法)和线程执行完成;ProgressChanged报告后台线程的事件,RunWorkerCompleted事件报告线程执行完毕的结果;简单的示例代码例子如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.ComponentModel;  
  3. using System.Threading;  
  4.   
  5. namespace ThreadDemo  
  6. {  
  7.     class BackgroudWorker0810  
  8.     {  
  9.        public static void Run()  
  10.         {  
  11.             TestBackgroudWorker();  
  12.             Console.ReadLine();  
  13.         }  
  14.         private static void TestBackgroudWorker()  
  15.         {  
  16.             Console.WriteLine("BackgroundWorker Begin");  
  17.             BackgroundWorker back = new BackgroundWorker();  
  18.             back.DoWork += new DoWorkEventHandler(BackgroundWorkerTest);  
  19.             back.RunWorkerAsync();  
  20.         }  
  21.         private static void BackgroundWorkerTest(object sender, DoWorkEventArgs es)  
  22.         {  
  23.             Thread.Sleep(1 * 1000);  
  24.             Console.WriteLine("Hello BackgroundWorker!");  
  25.         }  
  26.     }  
  27. }  
        包含完整功能的例子代码:

[csharp]  view plain  copy
  1. using System;  
  2. using System.ComponentModel;  
  3. using System.Threading;  
  4.   
  5. namespace ThreadDemo  
  6. {  
  7.     class BackgroudWorker0810Ex  
  8.     {  
  9.         private static BackgroundWorker back = new BackgroundWorker();  
  10.         public static void Run()  
  11.         {  
  12.             TestBackgroudWorker();  
  13.             Console.ReadLine();  
  14.         }  
  15.           
  16.         private static void TestBackgroudWorker()  
  17.         {  
  18.             back.WorkerSupportsCancellation = true;  
  19.             back.WorkerReportsProgress = true;  
  20.             back.DoWork += ExecuteMethod;  
  21.             back.ProgressChanged += BackProgressChanged;  
  22.             back.RunWorkerCompleted += BackWorkerCompleted;  
  23.             back.RunWorkerAsync();  
  24.   
  25.   
  26.             if (back.CancellationPending == false)  
  27.             {  
  28.                 Console.WriteLine("BackgroundWorker Still Run");  
  29.             }  
  30.             else  
  31.             {  
  32.                 Console.WriteLine("BackgroundWorker is Stop");  
  33.             }  
  34.             Thread.Sleep(5 * 1000);  
  35.             back.DoWork -= ExecuteMethod;  
  36.             back.CancelAsync();  
  37.             if (back.CancellationPending == false)  
  38.             {  
  39.                 Console.WriteLine("BackgroundWorker Still Run");  
  40.             }  
  41.             else  
  42.             {  
  43.                 Console.WriteLine("BackgroundWorker is Stop");  
  44.             }  
  45.   
  46.         }  
  47.         static void ExecuteMethod(object sender, DoWorkEventArgs es)  
  48.         {  
  49.             for (int i = 0; i < 20; i++)  
  50.             {  
  51.                 if (back.CancellationPending == false)  
  52.                 {  
  53.                     Thread.Sleep(1000);  
  54.                     back.ReportProgress(i, i * 100);  
  55.                 }  
  56.                 else  
  57.                 {  
  58.                     break;  
  59.                 }  
  60.             }  
  61.         }  
  62.         static void BackProgressChanged(object sender, ProgressChangedEventArgs e)  
  63.         {  
  64.             Console.WriteLine("UserState:"   
  65.                 + e.UserState.ToString()   
  66.                 + "***ProgressPercentage:"   
  67.                 + e.ProgressPercentage.ToString());  
  68.         }  
  69.         static void BackWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)  
  70.         {  
  71.             Console.WriteLine("BackWorkerCompleted");  
  72.         }  
  73.     }  
  74. }  

        3.互斥锁Monitor、Lock、Mutex

        用到多线程,常常就会牵扯到资源,此时就需要考虑线程安全性常用的是Monitor和Lock,其中Lock是C#的一个语法糖,Lock是对Monitor进行的封装,Monitor要注意的是操作过程中在finnally中一定要释放锁的资源(不能因为出异常而将资源一直锁着),Mutex与Monitor类似;线程安全性,就是在多线程情况下可能出现不一致的情况,简单的说就是多线程时执行结果的不确定性或者错误。
        不安全的示例代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class UnSafeCode0810  
  7.     {  
  8.        public static void Run()  
  9.         {  
  10.             TestUnSafe();  
  11.             Console.ReadLine();  
  12.         }  
  13.         private static int tempArg = 100;  
  14.         private static void TestUnSafe()  
  15.         {  
  16.             //两个线程都进行给TempArg加一的操作  
  17.             //两个线程同时进入到那段代码,实际值增加1(而不是2)  
  18.             ThreadPool.QueueUserWorkItem(new WaitCallback(AddValue));  
  19.             ThreadPool.QueueUserWorkItem(new WaitCallback(AddValue));  
  20.         }  
  21.         private static void AddValue(Object state)  
  22.         {  
  23.             Console.WriteLine("OldValue:" + tempArg.ToString());  
  24.             int i = tempArg;  
  25.             //这边实际有一系列的操作(这边就直接加一)              
  26.             i++;  
  27.             //耗时操作(这边就直接Sleep两秒)  
  28.             Thread.Sleep(2000);  
  29.             tempArg = i;  
  30.             Console.WriteLine("NewValue:" + tempArg.ToString());  
  31.         }  
  32.     }  
  33. }  
        Monitor示例代码:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class MonitorCode0810  
  7.     {  
  8.         private static int tempArg = 100;  
  9.         private static readonly object obj = new object();  
  10.         public static void Run()  
  11.         {  
  12.             TestMonitor();  
  13.             Console.ReadLine();  
  14.         }  
  15.   
  16.         private static void TestMonitor()  
  17.         {  
  18.             //两个线程都进行给TempArg加一的操作,由于方法里加了锁  
  19.             //解决了无锁时出现的问题,实际值增加2  
  20.             ThreadPool.QueueUserWorkItem(new WaitCallback(AddValue));  
  21.             ThreadPool.QueueUserWorkItem(new WaitCallback(AddValue));  
  22.         }  
  23.         private static void AddValue(Object state)  
  24.         {  
  25.             try  
  26.             {  
  27.                 Monitor.Enter(obj);  
  28.                 Console.WriteLine("OldValue:" + tempArg.ToString());  
  29.                 int i = tempArg;  
  30.                 //这边实际有一系列的操作(这边就直接加一)                  
  31.                 i++;  
  32.                 //耗时操作(这边就直接Sleep两秒)  
  33.                 Thread.Sleep(2000);  
  34.                 tempArg = i;  
  35.                 Console.WriteLine("NewValue:" + tempArg.ToString());  
  36.             }  
  37.             catch (Exception ex)  
  38.             {  
  39.                 Console.WriteLine(ex.ToString());  
  40.             }  
  41.             finally  
  42.             {  
  43.                 Monitor.Exit(obj);  
  44.             }  
  45.         }  
  46.     }  
  47. }  
        Lock示例代码:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class LockCode0810  
  7.     {  
  8.         public static void Run()  
  9.         {  
  10.             TestLock();  
  11.             Console.ReadLine();  
  12.         }  
  13.         private static int tempArg = 100;  
  14.         private static readonly object obj = new object();  
  15.         private static void TestLock()  
  16.         {  
  17.             //两个线程都进行给TempArg加一的操作  
  18.             //由于方法里加了锁,解决了无锁时出现的问题,实际值增加2  
  19.             ThreadPool.QueueUserWorkItem(new WaitCallback(AddValue));  
  20.             ThreadPool.QueueUserWorkItem(new WaitCallback(AddValue));  
  21.         }  
  22.         private static void AddValue(Object state)  
  23.         {  
  24.             lock (obj)  
  25.             {  
  26.                 Console.WriteLine("OldValue:" + tempArg.ToString());  
  27.                 int i = tempArg;  
  28.                 //这边实际有一系列的操作(这边就直接加一)  
  29.                 i++;  
  30.                 //耗时操作(这边就直接Sleep两秒)  
  31.                 Thread.Sleep(2000);  
  32.                 tempArg = i;  
  33.                 Console.WriteLine("NewValue:" + tempArg.ToString());  
  34.             }  
  35.         }  
  36.     }  
  37. }  
        Mutex示例代码:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class MutexCode0810  
  7.     {  
  8.         private static int tempArg = 100;  
  9.         private static Mutex mut = new Mutex();  
  10.         public static void Run()  
  11.         {  
  12.             TestMonitor();  
  13.             Console.ReadLine();  
  14.         }        
  15.   
  16.         private static void TestMonitor()  
  17.         {  
  18.             //两个线程都进行给TempArg加一的操作  
  19.             //由于方法里加了锁,解决了无锁时出现的问题,实际值增加2  
  20.             ThreadPool.QueueUserWorkItem(new WaitCallback(AddValue));  
  21.             ThreadPool.QueueUserWorkItem(new WaitCallback(AddValue));  
  22.         }  
  23.         private static void AddValue(Object state)  
  24.         {  
  25.             try  
  26.             {  
  27.                 mut.WaitOne();  
  28.                 Console.WriteLine("OldValue:" + tempArg.ToString());  
  29.                 int i = tempArg;  
  30.                 //这边实际有一系列的操作(这边就直接加一)  
  31.                 i++;  
  32.                 //耗时两秒(这边就直接Sleep两秒)  
  33.                 Thread.Sleep(2000);  
  34.                 tempArg = i;  
  35.                 Console.WriteLine("NewValue:" + tempArg.ToString());  
  36.             }  
  37.             catch (Exception ex)  
  38.             {  
  39.                 Console.WriteLine(ex.ToString());  
  40.             }  
  41.             finally  
  42.             {  
  43.                 mut.ReleaseMutex();  
  44.             }  
  45.         }  
  46.     }  
  47. }  

        4.线程池ThreadPool

        线程池,首先线程池是为了解决当程序有若干独立小任务执行时反复创建和销毁线程带来的性能损耗;将任务放到线程池中,线程执行完并不回收线程,降低了新建线程的开销,重复利用线程;默认情况下线程池会根据CPU的个数开辟相应的线程,其他的任务则在等待(例如4核CPU,12个线程请求进入,则会处理先进入的4个线程,另外8个则在等待之中,直到有线程执行完毕);如果同时有大量线程涌入,不会立即开启线程,而是等待一定时间间隔再新开一个线程(本机测试时是1s,本机是笔记本,cpu是i5,内存8g,64位系统),以此类推;如果要扩展默认同时处理的线程数可以设置指定个数的线程。示例代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class ThreadPool0810  
  7.     {  
  8.         public static void Run()  
  9.         {  
  10.             for (int i = 0; i < 20; i++)  
  11.             {  
  12.                 ThreadPool.QueueUserWorkItem(TestMethod, i);  
  13.             }  
  14.             Thread.Sleep(3 * 1000);  
  15.             Console.WriteLine("******LongTimeMethod******");  
  16.             for (int i = 0; i < 10; i++)  
  17.             {  
  18.                 ThreadPool.QueueUserWorkItem(LongTimeMethod, i);  
  19.             }  
  20.             Thread.Sleep(15 * 1000);  
  21.             Console.WriteLine("******LongTimeMethodEx******");  
  22.             ThreadPool.SetMinThreads(10, 0);  
  23.             for (int i = 0; i < 20; i++)  
  24.             {  
  25.                 ThreadPool.QueueUserWorkItem(LongTimeMethodEx, i);  
  26.             }  
  27.             Console.ReadLine();  
  28.         }  
  29.   
  30.         static void TestMethod(Object stateInfo)  
  31.         {  
  32.             Thread.Sleep(500);  
  33.             Console.WriteLine("****{0}****,{1}",  
  34.                 stateInfo.ToString(),   
  35.                 DateTime.Now.ToString("HH:mm:ss.fff"));  
  36.         }  
  37.         static void LongTimeMethod(Object stateInfo)  
  38.         {  
  39.             Console.WriteLine("****{0}****Begin****,{1}",  
  40.                 stateInfo.ToString(),  
  41.                 DateTime.Now.ToString("HH:mm:ss.fff"));  
  42.             Thread.Sleep(6 * 1000);  
  43.             Console.WriteLine("****{0}***End****,{1}",   
  44.                 stateInfo.ToString(),   
  45.                 DateTime.Now.ToString("HH:mm:ss.fff"));  
  46.         }  
  47.   
  48.         static void LongTimeMethodEx(Object stateInfo)  
  49.         {  
  50.             Console.WriteLine("****{0}****Begin****,{1}",  
  51.                 stateInfo.ToString(),  
  52.                 DateTime.Now.ToString("HH:mm:ss.fff"));  
  53.             Thread.Sleep(10 * 1000);  
  54.             Console.WriteLine("****{0}****End****,{1}",  
  55.                 stateInfo.ToString(),   
  56.                 DateTime.Now.ToString("HH:mm:ss.fff"));  
  57.         }  
  58.     }  
  59. }  

        5.线程间通信WaitHandle、ManualResetEvent、AutoResetEvent和Barrier;

        WaitHandle ,WaitAll等所有的都执行完,WaitAny等待最快的执行完。示例代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class WaitHandle0810  
  7.     {  
  8.         private static WaitHandle[] handles = new WaitHandle[]   
  9.                 {  
  10.                     new AutoResetEvent(false),  
  11.                     new AutoResetEvent(false),  
  12.                      new AutoResetEvent(false)  
  13.                 };  
  14.   
  15.         public static void Run()  
  16.         {  
  17.             Console.WriteLine("WaitAll Begin"  
  18.                 + DateTime.Now.ToString("hh:mm:ss.fff"));  
  19.             ThreadPool.QueueUserWorkItem(new WaitCallback(test1), handles[0]);  
  20.             ThreadPool.QueueUserWorkItem(new WaitCallback(test2), handles[1]);  
  21.             ThreadPool.QueueUserWorkItem(new WaitCallback(test3), handles[2]);  
  22.             WaitHandle.WaitAll(handles);  
  23.             Console.WriteLine("WaitAll End"   
  24.                 + DateTime.Now.ToString("hh:mm:ss.fff"));  
  25.   
  26.             Console.WriteLine("WaitAny Begin"   
  27.                 + DateTime.Now.ToString("hh:mm:ss.fff"));  
  28.             ThreadPool.QueueUserWorkItem(new WaitCallback(test1), handles[0]);  
  29.             ThreadPool.QueueUserWorkItem(new WaitCallback(test2), handles[1]);  
  30.             ThreadPool.QueueUserWorkItem(new WaitCallback(test3), handles[2]);  
  31.             WaitHandle.WaitAny(handles);  
  32.             Console.WriteLine("WaitAny End"   
  33.                 + DateTime.Now.ToString("hh:mm:ss.fff"));  
  34.             Console.ReadLine();  
  35.         }  
  36.         private static void test1(Object state)  
  37.         {  
  38.             AutoResetEvent are = (AutoResetEvent)state;  
  39.             Thread.Sleep(3 * 1000);  
  40.             are.Set();  
  41.         }  
  42.         private static void test2(Object state)  
  43.         {  
  44.             AutoResetEvent are = (AutoResetEvent)state;  
  45.             Thread.Sleep(4 * 1000);  
  46.             are.Set();  
  47.         }  
  48.         private static void test3(Object state)  
  49.         {  
  50.             AutoResetEvent are = (AutoResetEvent)state;  
  51.             Thread.Sleep(5 * 1000);  
  52.             are.Set();  
  53.         }  
  54.     }  
  55. }  
        ManualResetEvent,阻塞相关的所有线程,直至释放,释放时会释放所有的线程,释放完标志位无法自动重置(此时不再阻塞线程),需手动reset才可以继阻塞;示例代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class ManualResetEvent0810  
  7.     {  
  8.         static ManualResetEvent mre = new ManualResetEvent(false);  
  9.         public static void Run()  
  10.         {  
  11.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod01));  
  12.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod02));  
  13.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod03));  
  14.             Thread.Sleep(2 * 1000);  
  15.             Console.WriteLine("***************************************");  
  16.             mre.Set();  
  17.             Thread.Sleep(2 * 1000);  
  18.             Console.WriteLine("***************************************");  
  19.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod01));  
  20.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod02));  
  21.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod03));  
  22.             Thread.Sleep(2 * 1000);  
  23.             Console.WriteLine("***************************************");  
  24.             mre.Reset();  
  25.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod01));  
  26.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod02));  
  27.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod03));  
  28.             Thread.Sleep(2 * 1000);  
  29.             Console.WriteLine("***************************************");  
  30.             mre.Set();  
  31.             Thread.Sleep(2 * 1000);  
  32.             mre.Reset();  
  33.             Console.WriteLine("***************************************");  
  34.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod04));  
  35.             Console.ReadLine();  
  36.         }  
  37.         private static void TestMethod01(Object state)  
  38.         {  
  39.             Console.WriteLine("test1 Begin Wait,"  
  40.                 + DateTime.Now.ToString("hh:mm:ss"));  
  41.             mre.WaitOne();  
  42.             Console.WriteLine("test1 End Wait,"   
  43.                 + DateTime.Now.ToString("hh:mm:ss"));  
  44.         }  
  45.         private static void TestMethod02(Object state)  
  46.         {  
  47.             Console.WriteLine("test2 Begin Wait,"   
  48.                 + DateTime.Now.ToString("hh:mm:ss"));  
  49.             mre.WaitOne();  
  50.             Console.WriteLine("test2 End Wait,"   
  51.                 + DateTime.Now.ToString("hh:mm:ss"));  
  52.         }  
  53.         private static void TestMethod03(Object state)  
  54.         {  
  55.             Console.WriteLine("test3 Begin Wait,"   
  56.                 + DateTime.Now.ToString("hh:mm:ss"));  
  57.             mre.WaitOne();  
  58.             Console.WriteLine("test3 End Wait,"  
  59.                 + DateTime.Now.ToString("hh:mm:ss"));  
  60.         }  
  61.         private static void TestMethod04(Object state)  
  62.         {  
  63.             Console.WriteLine("TestMethod04 Begin Wait,"  
  64.                 + DateTime.Now.ToString("hh:mm:ss"));  
  65.             mre.WaitOne(2*1000);  
  66.             Console.WriteLine("TestMethod04 End Wait,"   
  67.                 + DateTime.Now.ToString("hh:mm:ss"));  
  68.         }  
  69.     }  
  70. }  
        AutoResetEvent,一次释放一个线程,且释放具有随机性(没有先后顺序),释放完会立即重置,为false时阻塞,为true时释放,释放完自动置为false;示例代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class AutoResetEvent0810  
  7.     {  
  8.         static AutoResetEvent autoResetEvent = new AutoResetEvent(false);  
  9.         public static void Run()  
  10.         {  
  11.             for (int i = 1; i <= 3; i++)  
  12.             {  
  13.                 Thread t = new Thread(TestMethod);  
  14.                 t.Name = "Thread" + i;  
  15.                 t.Start();  
  16.             }  
  17.             for (int i = 1; i <= 3; i++)  
  18.             {  
  19.                 Thread.Sleep(i * 1000);  
  20.                 autoResetEvent.Set();  
  21.             }  
  22.             Console.ReadLine();  
  23.         }  
  24.         private static void TestMethod()  
  25.         {  
  26.             string currentThreadName = Thread.CurrentThread.Name;  
  27.             Console.WriteLine("{0} Begin Wait,{1}",  
  28.                 currentThreadName,  
  29.                 DateTime.Now.ToString("hh:mm:ss.fff"));  
  30.             autoResetEvent.WaitOne();  
  31.             Console.WriteLine("{0} End Wait,{1}",   
  32.                 currentThreadName,  
  33.                 DateTime.Now.ToString("hh:mm:ss.fff"));  
  34.         }  
  35.     }  
  36.   
  37. }  
        Barrier,一次阻塞指定个数的线程(例如值为n),当有n个线程发出信号量时释放线程一次执行相应的操作,然后立即再次阻塞,直至又有累积n个线程发出信号量时才释放;这个思想在Log4net的数据库模式中有体现,log4net配置日志记录到数据库时可以设置当日志达到多少条时一次性插入。示例代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     public class Barrier0810  
  7.     {  
  8.         static Barrier barrier = new Barrier(2, (b) =>   
  9.         {   
  10.             Console.WriteLine("已经累积两个任务了,特此通告");   
  11.         });  
  12.         public static void Run()  
  13.         {  
  14.             ThreadPool.SetMinThreads(6, 0);  
  15.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod01));  
  16.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod02));  
  17.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod03));  
  18.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod04));  
  19.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod05));  
  20.             ThreadPool.QueueUserWorkItem(new WaitCallback(TestMethod06));  
  21.             Console.ReadLine();  
  22.         }  
  23.         private static void TestMethod01(Object state)  
  24.         {  
  25.             Thread.Sleep(1 * 1000);  
  26.             Console.WriteLine("test1"   
  27.                 + DateTime.Now.ToString("hh:mm:ss"));  
  28.             barrier.SignalAndWait();  
  29.         }  
  30.         private static void TestMethod02(Object state)  
  31.         {  
  32.             Thread.Sleep(2 * 1000);  
  33.             Console.WriteLine("test2"   
  34.                 + DateTime.Now.ToString("hh:mm:ss"));  
  35.             barrier.SignalAndWait();  
  36.         }  
  37.         private static void TestMethod03(Object state)  
  38.         {  
  39.             Thread.Sleep(3 * 1000);  
  40.             Console.WriteLine("test3"   
  41.                 + DateTime.Now.ToString("hh:mm:ss"));  
  42.             barrier.SignalAndWait();  
  43.         }  
  44.         private static void TestMethod04(Object state)  
  45.         {  
  46.             Thread.Sleep(1 * 1000);  
  47.             Console.WriteLine("TestMethod04"   
  48.                 + DateTime.Now.ToString("hh:mm:ss"));  
  49.             barrier.SignalAndWait();  
  50.         }  
  51.         private static void TestMethod05(Object state)  
  52.         {  
  53.             Thread.Sleep(2 * 1000);  
  54.             Console.WriteLine("TestMethod05"   
  55.                 + DateTime.Now.ToString("hh:mm:ss"));  
  56.             barrier.SignalAndWait();  
  57.         }  
  58.         private static void TestMethod06(Object state)  
  59.         {  
  60.             Thread.Sleep(3 * 1000);  
  61.             Console.WriteLine("TestMethod06"   
  62.                 + DateTime.Now.ToString("hh:mm:ss"));  
  63.             barrier.SignalAndWait();  
  64.         }  
  65.     }  
  66. }  

        6.任务Task、TaskFactory和Parallel    

        Task和TaskFactory,可以认为是线程和线程池的改良版(就跟Dictionary是HashTable的改良版一样),Task的内部采用线程池实现(针对多核进行了优化),Task可以中途取消,可以进行后继任务;默认情况Task是后台线程;Parallel对Task的一种封装,为常见循环操作 (For 循环、For each 循环)实现并行操作。

        证实Task内部采用线程池的代码:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading.Tasks;  
  3. using System.Threading;  
  4.   
  5. namespace ThreadDemo  
  6. {  
  7.     class Task0810  
  8.     {  
  9.         public static void Run()  
  10.         {  
  11.             ShowThreadPoolInfo("Task Begin");  
  12.             Task task1 = new Task(TestMethod);  
  13.             task1.Start();  
  14.             Task task2 = new Task(TestMethod);  
  15.             task2.Start();  
  16.             Task task3 = new Task(TestMethod);  
  17.             task3.Start();  
  18.             ShowThreadPoolInfo("Task End");  
  19.   
  20.             Console.ReadLine();  
  21.         }  
  22.         private static void ShowThreadPoolInfo(string info)  
  23.         {  
  24.             int workerThreads;  
  25.             int threads;  
  26.             ThreadPool.GetAvailableThreads(out workerThreads, out threads);  
  27.             Console.WriteLine("{0},workerThreads:{1},threads:{2},{3}",  
  28.                 info,   
  29.                 workerThreads,  
  30.                 threads,   
  31.                 DateTime.Now.ToString("HH:mm:ss.fff"));  
  32.         }  
  33.   
  34.         private static void TestMethod()  
  35.         {  
  36.             Thread.Sleep(1000);  
  37.             ShowThreadPoolInfo("TestMethod Run");  
  38.         }  
  39.     }  
  40. }  
        取消线程的示例代码:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3. using System.Threading.Tasks;  
  4.   
  5. namespace ThreadDemo  
  6. {  
  7.     class TaskCancel0810  
  8.     {  
  9.         private static CancellationTokenSource ct;  
  10.         public static void Run()  
  11.         {  
  12.             Console.WriteLine("Begin");  
  13.             ct = new CancellationTokenSource();  
  14.             Task task = new Task((() => TestMethod(ct)));  
  15.             task.Start();  
  16.             Thread.Sleep(5 * 1000);  
  17.             ct.Cancel();  
  18.             Console.ReadLine();  
  19.         }  
  20.         private static void TestMethod(CancellationTokenSource ct)  
  21.         {  
  22.             for (int i = 0; i < 10; i++)  
  23.             {  
  24.                 if (!ct.IsCancellationRequested)  
  25.                 {  
  26.                     Thread.Sleep(1 * 1000);  
  27.                     Console.WriteLine(DateTime.Now.ToString());  
  28.                 }  
  29.                 else  
  30.                 {  
  31.                     Console.WriteLine("Cancel");  
  32.                     break;  
  33.                 }  
  34.             }  
  35.         }  
  36.     }  
  37. }  
        后继任务、获得任务结果的示例代码:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading.Tasks;  
  3.   
  4. namespace ThreadDemo  
  5. {  
  6.     class TaskContinue0810  
  7.     {  
  8.         public static void Run()  
  9.         {  
  10.             Task<string> taskFirst = new Task<string>(TestFirst);  
  11.             taskFirst.Start();  
  12.             Task<string> taskContinue = taskFirst.ContinueWith((t) =>  
  13.             {  
  14.                 Console.WriteLine(t.Result);  
  15.                 return TestContinue();  
  16.             });  
  17.             taskContinue.ContinueWith((t) =>  
  18.             {  
  19.                 Console.WriteLine(t.Result);  
  20.             });  
  21.             Console.ReadLine();  
  22.         }  
  23.   
  24.         private static string TestFirst()  
  25.         {  
  26.             return "TestFirst";  
  27.         }  
  28.         private static string TestContinue()  
  29.         {  
  30.             return "TestContinue";  
  31.         }  
  32.     }  
  33. }  
        Parallel的示例代码:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading.Tasks;  
  3. using System.Threading;  
  4.   
  5. namespace ThreadDemo  
  6. {  
  7.     class Parallel0810  
  8.     {  
  9.         private static int N = 20;  
  10.         public static void Run()  
  11.         {  
  12.             Parallel.For(0, N, TestMethod);  
  13.             Console.ReadLine();  
  14.         }  
  15.         private static void TestMethod(int i)  
  16.         {  
  17.             Console.WriteLine(i.ToString() + "******"   
  18.                 + DateTime.Now.ToString("HH:mm:ss.fff"));  
  19.             Thread.Sleep(1 * 1000);  
  20.         }  
  21.     }  
  22. }  

        7.异步委托

        异步委托提供以异步方式调用同步方法的能力。示例代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Threading;  
  3. using System.Runtime.Remoting.Messaging;  
  4.   
  5. namespace ThreadDemo  
  6. {  
  7.     class AsyncResult0810  
  8.     {  
  9.         public static void Run()  
  10.         {  
  11.             IAsyncResult asyn1 = new Action(Task1)  
  12.                 .BeginInvoke(new AsyncCallback(Resule1), " 任务一");  
  13.             IAsyncResult asyn2 = new Func<int, Person>(Task2)  
  14.                 .BeginInvoke(100, new AsyncCallback(Resule2), " 任务二");  
  15.         }  
  16.         private static void Task1()  
  17.         {  
  18.             Thread.Sleep(1 * 1000);  
  19.             Console.WriteLine("任务一完成!");  
  20.         }  
  21.   
  22.         private static Person Task2(int value)  
  23.         {  
  24.             Thread.Sleep(2 * 1000);  
  25.             Console.WriteLine("任务二完成!");  
  26.             return new Person();  
  27.         }  
  28.   
  29.         /// <summary>  
  30.         /// 异步操作完成后调用的方法  
  31.         /// </summary>  
  32.         /// <param currentThreadName="result"></param>  
  33.         private static void Resule1(IAsyncResult result)  
  34.         {  
  35.             string msg = result.AsyncState.ToString();  
  36.             Console.WriteLine("这是耗时操作完成标志,额外信息是:{0}", msg);  
  37.             Console.WriteLine("没有异步返回值。");  
  38.         }  
  39.         /// <summary>  
  40.         /// 异步操作完成后调用的方法  
  41.         /// </summary>  
  42.         /// <param currentThreadName="result"></param>  
  43.         private static void Resule2(IAsyncResult result)  
  44.         {  
  45.             Func<int, Person> asyncDelegate   
  46.                 = (result as AsyncResult).AsyncDelegate as Func<int, Person>;  
  47.             string msg = result.AsyncState.ToString();  
  48.             ///person是异步返回的结果  
  49.             Person person = asyncDelegate.EndInvoke(result);  
  50.             Console.WriteLine("这是耗时操作完成标志,额外信息是:{0}", msg);  
  51.             Console.WriteLine("异步返回值:currentThreadName{0},age{1}",  
  52.                 person.name,  
  53.                 person.age);  
  54.         }  
  55.         /// <summary>  
  56.         /// 用户类(用于测试)  
  57.         /// </summary>  
  58.         private class Person  
  59.         {  
  60.             public string name = "pfe_Nova";  
  61.             public int age = 22;  
  62.         }  
  63.     }  
  64. }  

        8.其他,UI多线程,Timer等

        Winform中简单的常用Application.DoEvents()实现,WPF中常用Dispatcher.Run()实现;Timer一般用于定时周期性的操作,Timer示例代码如下:

[csharp]  view plain  copy
  1. using System;  
  2. using System.Timers;  
  3. using System.Threading;  
  4.   
  5. namespace ThreadDemo  
  6. {  
  7.     class Timer0810  
  8.     {  
  9.         static readonly System.Timers.Timer timer =  
  10.             new System.Timers.Timer();  
  11.         public static void Run()  
  12.         {  
  13.             TestTimer();  
  14.             Console.ReadLine();  
  15.         }  
  16.   
  17.         private static void TestTimer()  
  18.         {  
  19.             ShowThreadPoolInfo("TestTimer");  
  20.             timer.Enabled = false;  
  21.             SetInterval();  
  22.             timer.Elapsed += timerElapsed;  
  23.             timer.AutoReset = false;  
  24.             timer.Start();  
  25.             ShowThreadPoolInfo("TestTimer");  
  26.         }  
  27.         private static void SetInterval()  
  28.         {  
  29.             timer.Interval = 12 * 1000;  
  30.         }  
  31.         private static void timerElapsed(object sender, ElapsedEventArgs e)  
  32.         {  
  33.             Console.WriteLine("timerElapsed"   
  34.                 + DateTime.Now.ToString("hh:mm:ss.fff"));  
  35.             Thread.Sleep(5000);  
  36.             timer.Stop();  
  37.             SetInterval();  
  38.             timer.Start();  
  39.             Console.WriteLine("timerElapsed start"   
  40.                 + DateTime.Now.ToString("hh:mm:ss.fff"));  
  41.         }  
  42.         private static void ShowThreadPoolInfo(string info)  
  43.         {  
  44.             int workerThreads;  
  45.             int num;  
  46.             ThreadPool.GetAvailableThreads(out workerThreads, out num);  
  47.             Console.WriteLine("{0},workerThreads:{1},portThreads:{2},{3}",  
  48.                 info,  
  49.                 workerThreads,  
  50.                 num,  
  51.                 DateTime.Now.ToString("HH:mm:ss.fff"));  
  52.         }  
  53.     }  
  54. }  

        其他的诸如Thread.Sleep直接调用内核的指令,所在线程挂起,CPU执行队列的重排序,Thread.Sleep(0)的意义在于重新分配,而不是等待了。

        至此全部完成,只是个人的一些理解,对自己是一个记录,同时希望也能对别人有些帮助,如果有什么错误,还望不吝指教。转载请保留原文链接。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值