C#線程篇---Task(任務)和線程池
QueueUserWorkItem:沒有一個內建的機制讓你知道操作在什么時候完成,也沒有一個機制在操作完成是獲得一個返回值
static void Main(string[] args) { Console.WriteLine("主線程啟動"); //ThreadPool.QueueUserWorkItem(StartCode,5); new Task(StartCode, 5).Start(); Console.WriteLine("主線程運行到此!"); Thread.Sleep(1000); } private static void StartCode(object i) { Console.WriteLine("開始執行子線程...{0}",i); Thread.Sleep(1000);//模擬代碼操作 }
TaskCreationOptions這個類型是一個枚舉類型,傳遞一些標志來控制Task的執行方式(AttachedToParent標志,它總會得到Task采納,因為它和TaskScheduler本身無關)
static void Main(string[] args) { //1000000000這個數字會拋出System.AggregateException Task<Int32> t = new Task<Int32>(n => Sum((Int32)n), 1000000000); //可以現在開始,也可以以后開始 t.Start(); //Wait顯式的等待一個線程完成 t.Wait(); Console.WriteLine("The Sum is:"+t.Result); } private static Int32 Sum(Int32 i) { Int32 sum = 0; for (; i > 0; i--) checked { sum += i; } return sum; }
1. 在一個線程調用Wait方法時,系統會檢查線程要等待的Task是否已經開始執行,如果任務正在執行,那么這個Wait方法會使線程阻塞,知道Task運行結束為止。
2. 為什么要調用Wait或者Result?或者一直不查詢Task的Exception屬性?你的代碼就永遠注意不到這個異常的發生,如果不能捕捉到這個異常,垃圾回收時,拋出AggregateException,進程就會立即終止,這就是“牽一發動全身”,莫名其妙程序就自己關掉了,誰也不知道這是什么情況。所以,必須調用前面提到的某個成員,確保代碼注意到異常,並從異常中恢復。悄悄告訴你,其實在用Result的時候,內部會調用Wait。
三:任務工廠
Task還支持任務工廠的概念。任務工廠支持多個任務之間共享相同的狀態,如取消類型CancellationTokenSource就是可以被共享的。通過使用任務工廠,可以同時取消一組任務:
staticvoid Main(string[] args) { CancellationTokenSource cts =new CancellationTokenSource(); //等待按下任意一個鍵取消任務 TaskFactory taskFactory =new TaskFactory(); Task[] tasks =new Task[] { taskFactory.StartNew(() => Add(cts.Token)), taskFactory.StartNew(() => Add(cts.Token)), taskFactory.StartNew(() => Add(cts.Token)) }; //CancellationToken.None指示TasksEnded不能被取消 taskFactory.ContinueWhenAll(tasks, TasksEnded, CancellationToken.None); Console.ReadKey(); cts.Cancel(); Console.ReadKey(); } staticvoid TasksEnded(Task[] tasks) { Console.WriteLine("所有任務已完成!"); }
任務開始……
任務開始……
任務開始……
所有任務已完成(取消)!