亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

細說.NET中的多線程(三使用Task)

2019-11-14 13:53:13
字體:
來源:轉載
供稿:網友

 

上一節我們介紹了線程池相關的概念以及用法。我們可以發現ThreadPool. QueueUserWorkItem是一種起了線程之后就不管了的做法。但是實際應用過程,我們往往會有更多的需求,比如如果更簡單的知道線程池里面的某些線程什么時候結束,線程結束后如何執行別的任務。Task可以說是ThreadPool的升級版,在線程任務調度,并行編程中都有很大的作用。

創建并且初始化Task

使用lambda表達式創建Task

            Task.Factory.StartNew(() => Console.WriteLine("Hello from a task!"));            var task = new Task(() => Console.Write("Hello"));            task.Start();

  

用默認參數的委托創建Task

using System;using System.Threading.Tasks;namespace MultiThread{    class ThreadTest    {        static void Main()        {            var task = Task.Factory.StartNew(state => Greet("Hello"), "Greeting");            Console.WriteLine(task.AsyncState);   // Greeting            task.Wait();        }        static void Greet(string message) { Console.Write(message); }    }}

  

這種方式的一個優點是,task.AsyncState作為一個內置的屬性,可以在不同線程中獲取參數的狀態。

System.Threading.Tasks.TaskCreateOptions

創建Task的時候,我們可以指定創建Task的一些相關選項。在.Net 4.0中,有如下選項:

LongRunning

用來表示這個Task是長期運行的,這個參數更適合block線程。LongRunning線程一般回收的周期會比較長,因此CLR可能不會把它放到線程池中進行管理。

PReferFairness

表示讓Task盡量以公平的方式運行,避免出現某些線程運行過快或者過慢的情況。

AttachedToParent

表示創建的Task是當前線程所在Task的子任務。這一個用途也很常見。

下面的代碼是創建子任務的示例:

using System;using System.Threading;using System.Threading.Tasks;namespace MultiThread{    class ThreadTest    {        public static void Main(string[] args)        {            Task parent = Task.Factory.StartNew(() =>            {                Console.WriteLine("I am a parent");                Task.Factory.StartNew(() =>        // Detached task                {                    Console.WriteLine("I am detached");                });                Task.Factory.StartNew(() =>        // Child task                {                    Console.WriteLine("I am a child");                }, TaskCreationOptions.AttachedToParent);            });            parent.Wait();            Console.ReadLine();        }    }}

  

如果你等待你一個任務結束,你必須同時等待任務里面的子任務結束。這一點很重要,尤其是你在使用Continue的時候。(后面會介紹)

等待Task

在ThreadPool內置的方法中無法實現的等待,在Task中可以很簡單的實現了:

using System;using System.Threading;using System.Threading.Tasks;namespace MultiThread{    class ThreadTest    {        static void Main()        {            var t1 = Task.Run(() => Go(null));            var t2 = Task.Run(() => Go(123));            Task.WaitAll(t1, t2);//等待所有Task結束            //Task.WaitAny(t1, t2);//等待任意Task結束        }        static void Go(object data)   // data will be null with the first call.        {            Thread.Sleep(5000);            Console.WriteLine("Hello from the thread pool! " + data);        }    }}

  

注意:

當你調用一個Wait方法時,當前的線程會被阻塞,直到Task返回。但是如果Task還沒有被執行,這個時候系統可能會用當前的線程來執行調用Task,而不是新建一個,這樣就不需要重新創建一個線程,并且阻塞當前線程。這種做法節省了創建新線程的開銷,也避免了一些線程的切換。但是也有缺點,當前線程如果和被調用的Task同時想要獲得一個lock,就會導致死鎖。

Task異常處理

當等待一個Task完成的時候(調用Wait或者或者訪問Result屬性的時候),Task任務中沒有處理的異常會被封裝成AggregateException重新拋出,InnerExceptions屬性封裝了各個Task沒有處理的異常。

using System;using System.Threading.Tasks;namespace MultiThreadTest{    class Program    {        static void Main(string[] args)        {            int x = 0;            Task<int> calc = Task.Factory.StartNew(() => 7 / x);            try            {                Console.WriteLine(calc.Result);            }            catch (AggregateException aex)            {                Console.Write(aex.InnerException.Message);  // Attempted to divide by 0            }        }    }}

  

對于有父子關系的Task,子任務未處理的異常會逐層傳遞到父Task,并且最后包裝在AggregateException中。

using System;using System.Threading.Tasks;namespace MultiThreadTest{    class Program    {        static void Main(string[] args)        {            TaskCreationOptions atp = TaskCreationOptions.AttachedToParent;            var parent = Task.Factory.StartNew(() =>            {                Task.Factory.StartNew(() =>   // Child                {                    Task.Factory.StartNew(() => { throw null; }, atp);   // Grandchild                }, atp);            });            // The following call throws a NullReferenceException (wrapped            // in nested AggregateExceptions):            parent.Wait();        }    }}

  

取消Task

如果想要支持取消任務,那么在創建Task的時候,需要傳入一個CancellationTokenSouce

示例代碼:

using System;using System.Threading;using System.Threading.Tasks;namespace MultiThreadTest{    class Program    {        static void Main(string[] args)        {            var cancelSource = new CancellationTokenSource();            CancellationToken token = cancelSource.Token;            Task task = Task.Factory.StartNew(() =>            {                // Do some stuff...                token.ThrowIfCancellationRequested();  // Check for cancellation request                // Do some stuff...            }, token);            cancelSource.Cancel();            try            {                task.Wait();            }            catch (AggregateException ex)            {                if (ex.InnerException is OperationCanceledException)                    Console.Write("Task canceled!");            }            Console.ReadLine();        }    }}

  

任務的連續執行

Continuations

任務調度也是常見的需求,Task支持一個任務結束之后執行另一個任務。

            Task task1 = Task.Factory.StartNew(() => Console.Write("antecedant.."));            Task task2 = task1.ContinueWith(task =>Console.Write("..continuation"));

  

Continuations 和Task<TResult>

Task也有帶返回值的重載,示例代碼如下:

            Task.Factory.StartNew<int>(() => 8)                .ContinueWith(ant => ant.Result * 2)                .ContinueWith(ant => Math.Sqrt(ant.Result))                .ContinueWith(ant => Console.WriteLine(ant.Result));   // output 4

  

子任務

前面提到了,當你等待一個任務的時候,同時需要等待它的子任務完成。

下面代碼演示了帶子任務的Task:

using System;using System.Threading.Tasks;using System.Threading;namespace MultiThreadTest{    class Program    {        public static void Main(string[] args)        {            Task<int[]> parentTask = Task.Factory.StartNew(() =>            {                int[] results = new int[3];                Task t1 = new Task(() => { Thread.Sleep(3000); results[0] = 0; }, TaskCreationOptions.AttachedToParent);                Task t2 = new Task(() => { Thread.Sleep(3000); results[1] = 1; }, TaskCreationOptions.AttachedToParent);                Task t3 = new Task(() => { Thread.Sleep(3000); results[2] = 2; }, TaskCreationOptions.AttachedToParent);                t1.Start();                t2.Start();                t3.Start();                return results;            });            Task finalTask = parentTask.ContinueWith(parent =>            {                foreach (int result in parent.Result)                {                    Console.WriteLine(result);                }            });            finalTask.Wait();            Console.ReadLine();        }    }}

  

這段代碼的輸出結果是: 1,2,3

FinalTask會等待所有子Task結束后再執行。

TaskFactory

關于TaskFactory,上面的例子中我們使用了System.Threading.Tasks .Task.Factory屬性來快速的創建Task。當然你也可以自己創建TaskFactory,你可以指定自己的TaskCreationOptions,TaskContinuationOptions來使得通過你的Factory創建的Task默認行為不同。

.Net中有一些默認的創建Task的方式,由于TaskFactory創建Task的默認行為不同可能會導致一些不容易發現的問題。

如在.NET 4.5中,Task加入了一個Run的靜態方法:

Task.Run(someAction);

如果你用這個方法代替上面例子中的Task.Factory.StartNew,就無法得到正確的結果。原因是Task.Run創建Task的行為默認是默認是拒絕添加子任務的。上面的代碼等價于:

    Task.Factory.StartNew(someAction, CancellationToken.None, TaskCreationOptions.DenyChildAttach, TaskScheduler.Default);

你也可以創建具有自己默認行為的TaskFactory。

 

無論ThreadPool也好,或者Task,微軟都是在想進辦法來實現線程的重用,來節省不停的創建銷毀線程帶來的開銷。線程池內部的實現可能在不同版本中有不同的機制。如果可能的話,使用線程池來管理線程仍然是建議的選擇。

 

我們主要介紹了一下Task的基本用法,在我們編程過程中,有一些使用Task來提升程序性能的場景往往是很相似的,微軟為了簡化編程,在System.Threading.Tasks.Parallel中封裝了一系列的并行類,內部也是通過Task來實現的。

Parallel的For,Foreach,Invoke 方法

 

在編程過程中,我們經常會用到循環語句:

 

            for (int i = 0; i < 10; i++)            {                DoSomeWork(i);            }

  

如果循環過程中的工作可以是并行的話,那么我們可以用如下語句:

 

            Parallel.For(0, 10, i => DoSomeWork(i));

  

我們也經常會使用Foreach來遍歷某個集合:

 

            foreach (var item in collection)            {                DoSomeWork(item);            }

  

如果我們用一個線程池來執行里面的任務,那么我們可以寫成:

 

            Parallel.ForEach(collection, item => DoSomeWork(item));

  

最后,如果你想并行的執行幾個不同的方法,你可以:

 

            Parallel.Invoke(Method1, Method2, Method3);

  

如果你看下后臺的實現,你會發現基本都是基于Task的線程池,當然你也可以通過手動創建一個Task集合,然后等待所有的任務結束來實現同樣的功能。上面的Parallel.For和Parallel.Forach方法并不以為這你可以尋找你代碼里面所有用到For和Foreach方法,并且替代他們,因為每一個任務都會分配一個委托,并且在線程池里執行,如果委托里面的任務是線程不安全的,你可能還需要lock來保證線程安全,使用lock本身就會造成性能上的損耗。如果每一個任務都是需要長時間執行并且線程安全的,Parallel會給你帶來不錯的性能提升。對于短任務,或者線程不安全的任務,你需要權衡下,你是否真的需要使用Parallel。


作者:獨上高樓
出處:http://www.49028c.com/myprogram/
本文版權歸作者和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲白虎美女被爆操| 欧美激情xxxx| 成人国内精品久久久久一区| 日韩高清人体午夜| 国产亚洲aⅴaaaaaa毛片| 91亚洲国产成人精品性色| 国产69精品99久久久久久宅男| 久久久精品中文字幕| 狠狠色狠狠色综合日日小说| 久久91精品国产91久久久| 欧美限制级电影在线观看| 成人欧美一区二区三区黑人孕妇| 欧美激情综合色综合啪啪五月| 大伊人狠狠躁夜夜躁av一区| 日韩欧美在线视频观看| 欧美黑人一区二区三区| 亚洲一二三在线| 91最新国产视频| 日本老师69xxx| 亚洲一区二区免费| 91精品国产色综合久久不卡98| 这里只有精品在线观看| 成人性生交xxxxx网站| 亚洲一区二区三区乱码aⅴ| 亚洲男人天堂2019| 91精品国产自产在线老师啪| 久久五月情影视| 欧美精品中文字幕一区| 久久久午夜视频| 91网在线免费观看| 热99精品只有里视频精品| 91免费人成网站在线观看18| 伊人伊成久久人综合网小说| 色综合伊人色综合网| 欧美一级免费视频| 欧美日韩一区二区在线播放| 国产成人精品免费久久久久| 国产成人福利视频| 91久久精品视频| 精品日韩美女的视频高清| 国产在线精品成人一区二区三区| 亚洲精品久久久久久久久久久久久| 亚洲精品一区av在线播放| 欧美激情视频播放| 国产伦精品一区二区三区精品视频| 亚洲精品综合久久中文字幕| 成人激情av在线| 久久国产精品99国产精| 亚洲区一区二区| 伊人亚洲福利一区二区三区| 亚洲欧洲在线看| 91青草视频久久| 九九久久精品一区| 亚洲电影av在线| 97不卡在线视频| 久久国产精品电影| 亚洲男人的天堂在线播放| 国产成+人+综合+亚洲欧美丁香花| 国产精品午夜一区二区欲梦| 国产精品福利网站| 久久噜噜噜精品国产亚洲综合| 欧美黄色小视频| 国产a∨精品一区二区三区不卡| 国产精品最新在线观看| 欧美日韩国产91| 亚洲一级黄色av| 亚洲伦理中文字幕| 欧美精品video| 国内精品美女av在线播放| 51久久精品夜色国产麻豆| 日本一区二区三区四区视频| 精品少妇v888av| 96精品久久久久中文字幕| 亚洲欧洲xxxx| 国产精品高潮呻吟久久av野狼| 日韩中文字幕不卡视频| 成人精品福利视频| 日韩福利在线播放| 欧美性生活大片免费观看网址| 国产精品久久久久国产a级| 国模私拍视频一区| 亚洲国产高清福利视频| 日韩中文字幕不卡视频| 国产免费一区二区三区香蕉精| 国产精品wwwwww| 欧美成人激情图片网| 中文字幕亚洲一区二区三区五十路| 成人观看高清在线观看免费| 91欧美视频网站| 91网站免费看| 在线观看日韩www视频免费| 精品久久久久久久久中文字幕| 欧美性猛交xxxx乱大交蜜桃| 亚洲女同精品视频| 国产精品视频网| 中文字幕亚洲一区二区三区| 国产成人av在线播放| 狠狠色噜噜狠狠狠狠97| 国产精品18久久久久久首页狼| 亚洲一区二区三区sesese| 91av在线免费观看| 97人人爽人人喊人人模波多| 国产精品免费久久久久久| 国产在线视频一区| 欧美猛交ⅹxxx乱大交视频| 91精品视频在线播放| 欧美激情一区二区三区在线视频观看| 国产精品美女视频网站| 色偷偷91综合久久噜噜| 日韩美女av在线| 一本一本久久a久久精品综合小说| 一区二区三区美女xx视频| 57pao成人永久免费视频| 日韩亚洲第一页| 色噜噜久久综合伊人一本| 疯狂蹂躏欧美一区二区精品| 日韩中文字幕网址| 欧美天天综合色影久久精品| 自拍偷拍亚洲精品| 欧美中文在线字幕| 久久久久五月天| 91久久精品美女| 久久国产精品偷| 91精品视频在线| 亚洲最大福利视频网站| 久久久欧美一区二区| 国语对白做受69| 久久精品国产96久久久香蕉| 一区二区三区高清国产| 日韩风俗一区 二区| 18久久久久久| 美女久久久久久久久久久| 久久天堂av综合合色| 国产精品久久久久久久久久久不卡| 国内成人精品视频| 欧美激情伊人电影| 久久免费视频网| 欧美亚洲另类激情另类| 中文字幕日韩在线视频| 日韩av电影在线免费播放| 国产精品精品一区二区三区午夜版| 午夜美女久久久久爽久久| 国产极品jizzhd欧美| 久久免费观看视频| 国产偷国产偷亚洲清高网站| 亚洲欧美国产高清va在线播| 九九视频直播综合网| 色播久久人人爽人人爽人人片视av| 欧美另类老肥妇| 欧美一区二区三区艳史| 91成人天堂久久成人| 久久亚洲精品网站| 中文字幕日韩在线观看| 精品中文字幕乱| 日韩电影中文字幕| 久久伊人免费视频| 日韩欧美在线播放| 亚洲成人久久久久| 欧美亚洲在线视频| 国产一区在线播放| 国产精品永久免费在线| 欧美激情一二区| 欧美激情亚洲激情| 91欧美精品午夜性色福利在线|