<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
我們先來看下面的一個小范例:一個Winfrom程式,介面上有一個按鈕,有兩個非同步方法,點選按鈕呼叫兩個非同步方法,彈出執行順序,程式碼如下:
using System; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; namespace TPLDemoSln { public partial class Form1 : Form { public Form1() { InitializeComponent(); } /// <summary> /// 按鈕點選事件 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private async void btnStart_Click(object sender, EventArgs e) { string i1 = await F1Async(); MessageBox.Show("i1=" + i1); string i2 = await F2Async(); MessageBox.Show("i2=" + i2); } /// <summary> /// 非同步方法F1 /// </summary> /// <returns></returns> private Task<string> F1Async() { MessageBox.Show("F1 Start"); return Task.Run<string>(() => { // 休眠1秒 Thread.Sleep(1000); MessageBox.Show("F1 Run"); return "F1"; }); } /// <summary> /// 非同步方法F2 /// </summary> /// <returns></returns> private Task<string> F2Async() { MessageBox.Show("F2 Start"); return Task.Run<string>(() => { // 休眠2秒 Thread.Sleep(2000); MessageBox.Show("F2 Run"); return "F2"; }); } } }
在上面的程式碼中,Task.Run()是用來把一個程式碼段包裝為Task<T>的方法,Run中委託的程式碼體就是非同步任務執行的邏輯,最後return返回值。
執行程式,可以得到如下的輸出順序:
F1 Start->F1 Run->i1=F1->F2 Start->F2 Run->i2=F2。
我們對按鈕事件進行修改,修改為下面的程式碼,在看看執行順序:
/// <summary> /// 按鈕點選事件 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private async void btnStart_Click(object sender, EventArgs e) { //string i1 = await F1Async(); //MessageBox.Show("i1=" + i1); //string i2 = await F2Async(); //MessageBox.Show("i2=" + i2); Task<string> task1 = F1Async(); Task<string> task2 = F2Async(); string i1 = await task1; MessageBox.Show("i1=" + i1); string i2 = await task2; MessageBox.Show("i2=" + i2); }
再次執行程式,檢視輸出順序:
F1 Start->F2 Start->F1 Run->i1=F1->F2 Run->i2=F2。
可以看出兩次的執行順序不一致。這是什麼原因呢?
這是因為並不是到了await才開始執行Task非同步任務,執行到Task<string> task1=F1Async()這句程式碼的時候,F1Async非同步任務就開始執行了。同理,執行到下一句程式碼就開始執行F2Async非同步任務了。await是為了保證執行到這裡的時候非同步任務一定執行完。執行到await的時候,如果非同步任務還沒有執行,那麼就等待非同步任務執行完。如果非同步任務已經執行完了,那麼就直接獲取非同步任務的返回值。
我們上面解釋的原因是否正確呢?我們可以做下面的一個實驗,來驗證上面說的原因,我們把按鈕事件裡面的await註釋掉,看看非同步方法還會不會執行,程式碼如下:
/// <summary> /// 按鈕點選事件 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private async void btnStart_Click(object sender, EventArgs e) { //string i1 = await F1Async(); //MessageBox.Show("i1=" + i1); //string i2 = await F2Async(); //MessageBox.Show("i2=" + i2); Task<string> task1 = F1Async(); Task<string> task2 = F2Async(); // 並不是到了await才開始執行Task非同步任務,這裡是保證非同步任務一定執行完 // 程式碼執行到這裡的時候,如果沒有執行完就等待執行完,如果已經執行完,就直接獲取返回值 // 這裡註釋掉await程式碼段,檢視非同步方法是否會執行 //string i1 = await task1; //MessageBox.Show("i1=" + i1); //string i2 = await task2; //MessageBox.Show("i2=" + i2); }
執行程式,發現非同步方法還是會執行,這就說明我們上面解釋的原因是正確的。感興趣的可以使用Reflector反編譯檢視內部實現的原理,主要是MoveNext()方法內部。
我們可以得到下面的結論:
上面說的第二點看下面的程式碼
/// <summary> /// 方法標記為async 直接返回一個int型別的數值即可 /// </summary> /// <returns></returns> private async Task<int> F3Async() { return 2; }
上面的第三點可以看下面的程式碼:
/// <summary> /// 方法沒有被標記為async,直接返回一個Task /// </summary> /// <returns></returns> private Task<int> F4Async() { return Task.Run<int>(() => { return 2; }); }
我們做一些總結:
1、如果方法內部有await,則方法必須標記為async。await和async是成對出現的,只有await沒有async程式會報錯。只有async沒有await,程式會按照同步方法執行。
2、ASP.NET MVC中的Action方法和WinForm中的事件處理方法都可以標記為async,控制檯的Main()方法不能被標記為async。對於不能標記為async的方法怎麼辦呢?我們可以使用Result屬性來獲取值,看下面程式碼:
using System; using System.Net.Http; using System.Threading.Tasks; namespace AsyncDemo { class Program { static void Main(string[] args) { // 範例化物件 HttpClient client = new HttpClient(); // 呼叫非同步的Get方法 Task<HttpResponseMessage> taskMsg = client.GetAsync("http://www.baidu.com"); // 通過Result屬性獲取返回值 HttpResponseMessage msg = taskMsg.Result; Task<string> taskRead = msg.Content.ReadAsStringAsync(); string html = taskRead.Result; Console.WriteLine(html); Console.ReadKey(); } } }
不建議使用這種方式,這樣體現不出非同步帶來的好處,而且使用Result屬性,有可能會帶來上下文切換造成的死鎖。
下面我們來看看建立Task的方法。
1、如果返回值就是一個立即可以隨手得到的值,那麼就用Task.FromResult()。看下面程式碼:
static Task<int> TestAsync() { //return Task.Run<int>(() => //{ // return 5; //}); // 簡便寫法 return Task.FromResult(3); }
2、如果是一個需要休息一會的任務(比如下載失敗則過5秒鐘後重試。主執行緒不休息,和Thread.Sleep不一樣),那麼就用Task.Delay()。
3、Task.Factory.FromAsync()會把IAsyncResult轉換為Task,這樣APM風格的API也可以用await來呼叫。
4、編寫非同步方法的簡化寫法。如果方法宣告為async,那麼可以直接return具體的值,不用在建立Task,由編譯器建立Task,看下面的程式碼:
static async Task<int> Test2Async() { // 複雜寫法 //return await Task.Run<int>(() => //{ // return 5; //}); // 下面是簡化寫法,直接返回 return 6; }
到此這篇關於C#多執行緒TPL模式高階用法探祕的文章就介紹到這了。希望對大家的學習有所幫助,也希望大家多多支援it145.com。
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45