聊聊多線程哪一些事兒(task)之 三 非同步取消和非同步方法

来源:https://www.cnblogs.com/xiaoXuZhi/archive/2019/12/30/XYH_tsak_WhenAny.html
-Advertisement-
Play Games

介紹了task的創建、運行、同步/非同步執行、阻塞等待,本文將主要介紹task的延續操作:WhenAny、WhenAll、ContinueWith;CancellationTokenSource任務取消;非同步方法之:(async/await) ...


   hello,咋們又見面啦,通過前面兩篇文章的介紹,對task的創建、運行、阻塞、同步、延續操作等都有了很好的認識和使用,結合實際的場景介紹,這樣一來在實際的工作中也能夠解決很大一部分的關於多線程的業務,但是只有這一些是遠遠不夠的,比如,比如,如果這麼一個場景,當開啟tsak非同步任務後,有某個條件觸發,需要終止tsak的執行又該如何實現呢?這一些問題正是我們今天需要交流分享的部分,帶著這一些問題,咱們共同進入到今天的主題,謝謝!

    在進入主題前,如果你沒有閱讀前面的兩篇文章,歡迎您點擊下麵地址先閱讀一下,這樣能夠更加連貫的掌握瞭解今天的內容,謝謝!

  第一篇:聊聊多線程哪一些事兒(task)之 一創建運行與阻塞

  第二篇:聊聊多線程哪一些事兒(task)之 二 延續操作

  第三篇:聊聊多線程哪一些事兒(task)之 三 非同步取消和非同步方法

 

Task之任務取消:CancellationTokenSource

 

    關於線程取消,我相信大家在實際工作中都會遇到這樣的問題,無論是採用哪一種方式實現非同步線程,都會有相應的機制來取消線程操作。本次將同時對Thread的線程取消實現,Tsak的線程取消實現同時通過實例說明。

    在我的工作經驗中,需要取消非同步線程作業的實際使用場景往往是一些非同步作業程式,也就是一些周期性的,迴圈業務操作。比如周期性的數據同步、數據更新等等操作。比如:電商系統常見的一個場景,訂單超時取消等等。

    為了與前兩篇的實例保持一致性,我現在還是以酒店平臺的數據同步業務為例:

    需求:每周三凌晨3點鐘,通過攜程提供的酒店分頁查詢介面,全量同步一次最新的酒店數據。並且能夠通過人為的干預來終止數據同步操作。

    下麵我將分別通過Thread和task兩種方式來實現

    其一、Thread時代之任務取消

    哈哈,實話實話說,在幾年前的項目中,我也是採用Thread來實現非同步線程的,也會遇到線程的取消的業務場景。我當時的實現方式是,定義一個全局變數,isStopThread(是否終止線程),去過需要取消任務,只需要控制isStopThread的值即可,每一次執行具體的業務時,首先判斷一下isStopThread,只有非終止狀態才執行具體的業務邏輯。

 /// <summary>
 /// 攜程 酒店數據同步作業(Thread)
 /// </summary>
 private static void CtripHoteDataSynchrByThread()
 {
     CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
     // 第一步通過thread開啟一個線程
     Thread thread = new Thread(() =>
     {
         // 獲取數據的次數
         int getDataIndex = 1;
         isStopCtripHoteDataSynchr = false;

         // 通過調用攜程的分頁服務,獲取其有效的酒店數據
         // 在獲取數據前,首先判斷一下是否終止獲取

         while (!cancellationTokenSource.IsCancellationRequested)
         {
             // 現在假設模擬,獲取攜程的所有有效的酒店數據通過 3 次就獲取完畢
             Console.WriteLine($"開始獲取攜程第 {getDataIndex} 頁酒店數據....\n");

             Thread.Sleep(3000);
             Console.WriteLine($"攜程第 {getDataIndex} 頁酒店數據獲取完畢\n");
             getDataIndex++;

             // 模擬獲取完第三頁數據,代表數據獲取完畢,直接終止掉
             if (getDataIndex == 4)
             {
                 Console.WriteLine($"同步完畢攜程的所有酒店數據\n");
                 break;
             }
         }

         if (isStopCtripHoteDataSynchr)
         {
             Console.WriteLine($"取消同步攜程酒店數據\n");
         }
     });

     thread.Start();

     Console.WriteLine("攜程酒店數據同步中.....\n");
     // 模擬實際數據同步中的取消操作
     Console.WriteLine("如果需要取消數據同步,那麼請輸入任意字元即可取消操作\n");

     Console.ReadLine();

     cancellationTokenSource.Cancel();
     isStopCtripHoteDataSynchr = true;

     Console.WriteLine($"發起取消同步攜程酒店數據請求\n");
 }

執行結果:

    通過測試結果我們可以看到,在獲取第2頁數據時,此時發起了一個取消線程命令,當第二頁數據獲取完畢後,線程就裡面終止了,從而到達了線程取消的目的。

    其二、Task時代之任務取消

    隨著Task的推出,微軟也推出了一個專門服務於線程取消的幫助類(CancellationTokenSource),通過該類能夠很好的幫助我們取消一個線程,話不多說,我們先通過CancellationTokenSource類實現上面示例的功能。

/// <summary>
 /// 攜程 酒店數據同步作業(Task)
 /// </summary>
 private static void CtripHoteDataSynchrByTask()
 {
     // 定義任務取消機制
     CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();

     // 第一步通過thread開啟一個線程
     Task thread = new Task(() =>
     {
         // 獲取數據的次數
         int getDataIndex = 1;

         // 通過調用攜程的分頁服務,獲取其有效的酒店數據
         // 在獲取數據前,首先判斷一下是否終止獲取

         while (!cancellationTokenSource.IsCancellationRequested)
         {
             // 現在假設模擬,獲取攜程的所有有效的酒店數據通過 3 次就獲取完畢
             Console.WriteLine($"開始獲取攜程第 {getDataIndex} 頁酒店數據....\n");

             Console.WriteLine($"攜程第 {getDataIndex} 頁酒店數據獲取完畢\n");
             getDataIndex++;

             // 模擬獲取完第三頁數據,代表數據獲取完畢,直接終止掉
             if (getDataIndex == 4)
             {
                 Console.WriteLine($"同步完畢攜程的所有酒店數據\n");
                 break;
             }
         }

         if (cancellationTokenSource.IsCancellationRequested)
         {
             Console.WriteLine($"取消同步攜程酒店數據\n");
         }
     });
     thread.Start();

     Console.WriteLine("攜程酒店數據同步中.....\n");
     // 模擬實際數據同步中的取消操作
     Console.WriteLine("如果需要取消數據同步,那麼請輸入任意字元即可取消操作\n");

     Console.ReadLine();

     // 直接取消線程
     cancellationTokenSource.Cancel();

     // 在指定時間後取消線程
     // cancellationTokenSource.CancelAfter(1000);

     Console.WriteLine($"發起取消同步攜程酒店數據請求\n");
 }
 

測試結果:

 

    通過測試結果,兩種實現方式的結果完全一致

    當然,CancellationTokenSource 還提供了CancelAfter(多久後取消)方法,來實現多久後取消線程。

    說到這兒,不知道大家有沒有發現一個問題CancellationTokenSource 其實現是不是與Task和Thread沒有多少關係,在第一個實例中通過Thread實現的線程取消,同樣可以結合CancellationTokenSource 來實現。所以說,在開始我說CancellationTokenSource 是微軟提供的一個線程取消的一個幫助類就是這個原因。其實我可以打開CancellationTokenSource 的實現源碼,其實我們就會一目瞭然,其取消線程的核心邏輯和我們上面的說Thread取消的原理很類似,都是控制一個變數的值來實現,只是CancellationTokenSource 對其所有操作進行了一個封裝。其中的CancelAfter裡面是開啟了一個定義器,定時器的最終實現還是和Canel一樣。如果想看CancellationTokenSource的源碼,大家可以查看下麵地址:https://www.cnblogs.com/majiang/p/7920102.html

最後需要說明的是Task與CancellationTokenSource都是.net Framework4.0+、.NET Core、.NET Standard。

 

非同步方法之:(async/await)

 

    c#5.0微軟推出了一個新的特性那就是非同步方法,其關鍵詞為async。有了async我們要實現一個非同步方法就簡單的多啦,你會發現和實現一個同步方法很相似,只需要對方法加以async修飾即可。當然如果只是簡單的修飾調用,那麼也會是同步調用,為了達到真正的非同步調用,往往是需要另外一個關鍵詞await來配合使用。

    先簡單介紹一下async非同步函數:

    async的三種返回類型:

    Tsak:其主要適用場景是,主程式只關心非同步方法執行狀態,不需要和主線程有任何執行結果數據交互。

    Task<T>:其主要適用場景是,主程式不僅僅關心非同步方法執行狀態,並且還希望執行後返回一個數據類型為T的結果

    void: 主程式既不關係非同步方法執行狀態,也不關心其執行結果,只是主程式調用一次非同步方法,對於除事件處理程式以外的代碼,通常不鼓勵使用 async void 方法,因為調用方不能

    在介紹一下await關鍵詞:

    await其顧名思義就是等待的意思,其運行原理就是:調用方執行到await時就會立即返回,但是非同步方法等待非同步執行結果。所以await只能存在於async修飾的非同步方法體中,await不阻塞主線程,只是阻塞當前非同步方法繼續往下執行,這樣就能夠達到真正非同步的目的。

下麵以一個簡單的例子來說明一下每一種情況的使用:

static void Main(string[] args)
 {
     Console.WriteLine("主線程開始\n");
     Console.WriteLine("主線程調用同步方法:SynTest\n");
     SynTest();

     Console.WriteLine("主線程調用非同步方法:AsyncTestNoAwait\n");
     AsyncTestNoAwait();

     Console.WriteLine("主線程調用非同步方法:AsyncTestHasAwait\n");
     AsyncTestHasAwait();

     Console.WriteLine("主線程結束\n");
     Console.ReadKey();
 }

 /// <summary>
 /// 同步方法測試
 /// </summary>
 public static void SynTest()
 {
     Console.WriteLine("同步方法SynTest開始運行\n");
     Thread.Sleep(5000);
     Console.WriteLine("同步方法SynTest運行結束\n");
 }

 /// <summary>
 /// 非同步方法測試(不帶有 await關鍵詞)
 /// </summary>
 public static async void AsyncTestNoAwait()
 {
     Console.WriteLine("非同步方法AsyncTestNoAwait開始運行\n");
     Thread.Sleep(5000);
     Console.WriteLine("非同步方法AsyncTestNoAwait運行結束\n");
 }

 /// <summary>
 /// 非同步方法測試(帶有 await關鍵詞)
 /// </summary>
 public static async void AsyncTestHasAwait()
 {
     Console.WriteLine("非同步方法AsyncTestHasAwait開始運行\n");
     await Task.Delay(5000);
     Console.WriteLine("非同步方法AsyncTestHasAwait運行結束\n");
 }

運行結果:

從運行結果我們可以很好的得出:

    1、非同步方法async如果沒有await關鍵詞,其執行原理還是同步調用

    2、await關鍵詞只能存在雲async修飾的方法體中

    3、非同步方法async在調用時,只有遇到await關鍵詞後的程式塊才是非同步執行,其await關鍵詞前的代碼塊還是同步執行

    好了,管理async先介紹到這兒,由於時間和文章篇幅原因,就不在詳細介紹,裡面還有很多內容需要註意,後續在根據實際做一個async/await的專題文章。

總結:

    到目前為止,有關Task的3篇文章都到此結束,下麵在回顧總結一下Task的相關功能點吧!

    1、Task的創建運行可以有三種方式:new Task/Task.Factory/Task.Run

    2、Task的返回參數定義Task<返回類型>

        獲取返回值:Task.Result->要阻塞主流程

    3、Task線程的同步實現不僅僅可以通過RunSynchronously來實現同步運行,當然還可以通過Task.Result/Task.Wait等方式來變向實現

    4、Task的wait/waitAll/waitAny實現阻塞等待執行結果

    5、Task的WhenAny、WhenAll、ContinueWith實現延續操作

    6、CancellationTokenSource實現非同步任務取消

    7、非同步方法之:(async/await)實現同步和非同步調用等

 

猜您喜歡: 

 第一篇:聊聊多線程哪一些事兒(task)之 一創建運行與阻塞

 第二篇:聊聊多線程哪一些事兒(task)之 二 延續操作

END
為了更高的交流,歡迎大家關註我的公眾號,掃描下麵二維碼即可關註,謝謝:


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 目 錄 1. 概述... 2 2. 演示信息... 2 3. iNeuView(Web組態)配置數據介面... 2 4. iNeuView(Web組態)圖元綁定數據... 4 5. iNeuView(Web組態)圖元和文本框配置預警... 5 6. iNeuView(Web組態)圖元和文本框自定義右 ...
  • Wei.Repository 基於EFCore3.0+Dapper 封裝Repository,實現UnitOfWork,提供基本的CURD操作,可直接註入泛型Repository,也可以繼承Repository,重寫CURD操作 Githut:https://github.com/a34546/We ...
  • 0.準備工作:IIS6.0鏡像包,自製的網頁文件夾(路徑不能是桌面,否則其他電腦將因為沒有許可權訪問系統桌面而不能訪問你的網頁) 1.進入添加或刪除程式,勾上Internet信息服務(IIS),點擊下一步,將文件路徑選擇為IIS6.0解壓後文件(需要確認多次,請等待),安裝完成。 2.在管理工具中找到 ...
  • 本筆記摘抄自:https://www.cnblogs.com/ArmyShen/archive/2012/08/27/2659405.html,記錄一下學習過程以備後續查用。 索引器允許類或者結構的實例按照與數組相同的方式進行索引取值,索引器與屬性類似,不同的是索引器的訪問是帶參的。 索引器和數組比 ...
  • 1、centos7 上安裝.net core sdk 3.1 參考官方所給的教程 官方建議直接安裝運行時 我安裝了sdk 發現也安裝了依賴項運行時 https://docs.microsoft.com/zh cn/dotnet/core/install/linux package manager c ...
  • AOP(面向切麵編程),通過預編譯方式和運行期間動態代理實現程式功能的統一維護的一種技術。AOP是OOP的延續,是函數式編程的一種衍生範型。利用AOP可以對業務邏輯的各個部分進行隔離,從而使得業務邏輯各部分之間的耦合度降低,提高程式的可重用性,同時提高了開發的效率。 AspectCore 提供了一個 ...
  • 原文:https://www.stevejgordon.co.uk/httpclientfactory-using-polly-for-transient-fault-handling發表於:2018年6月 原文:https://www.stevejgordon.co.uk/httpclientfa ...
  • 本筆記摘抄自:https://www.cnblogs.com/solan/archive/2012/08/01/CSharp06.html,記錄一下學習過程以備後續查用。 摘要: 抽象類:是一種特殊的類,可以定義具有實現的方法,也可以定義未實現的方法契約,本身不能被實例化,只能在派生類中進行實例化。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...