主頁 > 作業系統 > 如何按順序等待C#任務,而不是并行等待?

如何按順序等待C#任務,而不是并行等待?

2021-10-16 16:56:56 作業系統

我有一組在外部硬體上運行的異步測驗。

我可以按順序運行它們,但因為所有這些測驗都有副作用,我希望能夠對它們進行洗牌并一遍又一遍地運行它們。

當我將它們放在一個串列中并嘗試在之后等待每個時,它們都并行運行,而不是 1 個運行。

我希望能夠洗牌串列,然后按順序運行它們。

我如何List<Task<bool>按順序運行測驗 ( ),同時將它們保存在串列中,以便我可以隨機播放串列?

private async void RunVisca()
{
    Ptz ptz = new Ptz(SelectedComPort, (byte)(1), SelectedBaudRate, Node.DeviceType.PTZ);

    UpdateResults("VISCA TEST START\n\n");

    // Method 1: Correct - Runs in order
    await VTestDriveClosedLoop(ptz, true);
    await VTestDriveClosedLoop(ptz, false);
    await VTestLimitsCl(ptz, 125, 20);
    await VTestLimitsOl(ptz, 125, 20);

    // Method 2: Incorrect - Runs in parallel
    List<Task<bool>> tests = new List<Task<bool>>();
    tests.Add(VTestDriveClosedLoop(ptz, true));
    tests.Add(VTestDriveClosedLoop(ptz, false));
    tests.Add(VTestLimitsCl(ptz, 125, 20));
    tests.Add(VTestLimitsOl(ptz, 125, 20));

    // Tasks all run in parallel here - should they not run in order?
    foreach(Task<bool> test in tests)
    {
        _ = await test.ConfigureAwait(true);
    }

    UpdateResults("\nTEST COMPLETE\n\n");
}

uj5u.com熱心網友回復:

基于任務的異步模式 (TAP) 模式意味著呼叫async回傳 a方法Task將回傳熱任務

Task處于熱狀態的A已經啟動。這可以通過檢查Task.Status屬性來證明,該屬性永遠不會是TaskStatus.Created

一個寒冷的任務將有一個Task.Status財產TaskStatus.Created和才會開始運行的時候Start()被呼叫的實體。

創建冷任務的唯一方法是使用Task/Task<T>各自的公共建構式。


由于您的任務是通過異步方法回傳的,因此當它們回傳給您時它們已經在運行。

要按順序而不是并行運行它們,除非前一個任務已完成,否則您不能一個接一個地創建它們。為確保這一點,您必須分別處理await它們。

這導致方法 1 是按順序而不是并行運行熱任務的正確方法

await VTestDriveClosedLoop(ptz, true);
await VTestDriveClosedLoop(ptz, false);
await VTestLimitsCl(ptz, 125, 20);
await VTestLimitsOl(ptz, 125, 20);

方法 2/3 不正確,因為它將并行運行您的任務。

您正在將從您的async方法回傳的熱門任務添加到您的串列中,而無需等待它們。在您繼續將下一個任務添加到串列之前,它們將已經開始運行

第一任務是不等待,因此第二次任務后立即啟動,不管任務完成1與否。

不等待第二個任務,所以第三個任務等。

使用方法 2/3,它們將始終并行運行。

tests.Add(VTestDriveClosedLoop(ptz, true));
tests.Add(VTestDriveClosedLoop(ptz, false));

真正的問題是如何按順序運行冷任務,以便我們可以將它們存盤在串列中,同時利用延遲執行,直到我們遍歷我們的串列。

解決方案是存盤委托,這將觸發以下任務建構式:

public Task (Func<TResult> function);

這將創建冷任務,讓您TaskFunc<Task<bool>>呼叫運行

var tests = new List<Func<Task<bool>>>();
tests.Add(() => VTestDriveClosedLoop(ptz, true));
tests.Add(() => VTestDriveClosedLoop(ptz, false));
tests.Add(() => VTestLimitsCl(ptz, 125, 20));
tests.Add(() => VTestLimitsOl(ptz, 125, 20));
    
foreach (var test in tests) {
     await test();
}

下面的演示程式將清楚地顯示按順序運行與并行運行的任務。

class Program
{
    private static async Task Main(string[] args)
    {
        // 1... 2... 3... 4...
        var tasksThatRunInOrder = new List<Func<Task<bool>>>();
        tasksThatRunInOrder.Add(() => Test("1"));
        tasksThatRunInOrder.Add(() => Test("2"));
        tasksThatRunInOrder.Add(() => Test("3"));
        tasksThatRunInOrder.Add(() => Test("4"));

        foreach (var test in tasksThatRunInOrder)
            await test();

        // 1, 2, 3, 4
        var testsThatRunInParallel = new List<Task<bool>>();
        testsThatRunInParallel.Add(Test("1"));
        testsThatRunInParallel.Add(Test("2"));
        testsThatRunInParallel.Add(Test("3"));
        testsThatRunInParallel.Add(Test("4"));

        foreach (var test in testsThatRunInParallel)
            await test;
    }

    private static async Task<bool> Test(string x)
    {
        Console.WriteLine(x);
        await Task.Delay(1000);
        return true;
    }
}

uj5u.com熱心網友回復:

任務在創建時開始。由于您是同時創建所有這些而不等待它們中的任何一個,因此它們將以不確定的并行方式執行。我想也許您想改為存盤代表。這些在被呼叫之前不會執行。

var tests = new List<Func<Task<bool>>>();
tests.Add( () => VTestDriveClosedLoop(ptz, true) );
tests.Add( () => VTestDriveClosedLoop(ptz, false) );
tests.Add( () => VTestLimitsCl(ptz, 125, 20) );
tests.Add( () => VTestLimitsOl(ptz, 125, 20) );
    
foreach(var test in tests)
{
    _ = await test(); 
}

uj5u.com熱心網友回復:

您可以創建一個串列,在其中隨機化要運行的順序,然后相應地呼叫創建任務的函式。例如:

    static async Task RunTasks()
    {
        Random rng = new Random();
        var order = new List<int>() { 1, 2 };
        var shuffledOrder = order.OrderBy(a => rng.Next()).ToList();

        foreach (var i in shuffledOrder)
        {
            switch (i)
            {
            case 1:
                await LoadFile1();
                break;
            case 2:
                await LoadFile2();
                break;
            }
        }
    }

uj5u.com熱心網友回復:

嘗試在不分配給一次性變數的情況下等待。

uj5u.com熱心網友回復:

的呼叫鏈怎么樣ContinueWith

Task.Run(async () => 
{
    await VTestDriveClosedLoop(ptz, true);
}).ContinueWith(async (_) => 
{ 
    await VTestDriveClosedLoop(ptz, false);
}).ContinueWith(async (_) => 
{ 
    await VTestLimitsCl(ptz, 125, 20);
}).ContinueWith(async (_) => 
{ 
    await VTestLimitsOl(ptz, 125, 20);
});

轉載請註明出處,本文鏈接:https://www.uj5u.com/caozuo/318807.html

標籤:C# 。网 异步 异步等待 任务

上一篇:如何使node.js功能異步?有時仍然回傳空物件

下一篇:AsyncFunction().GetAwaiter().GetResult()和Task.Run(()=>AsyncFunction).GetAwaiter().GetResult()有什么區

標籤雲
其他(123570) Java(13369) Python(12731) C(7545) 區塊鏈(7372) JavaScript(7059) 基礎類(6313) AI(6244) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4120) MySQL(4012) Linux(3394) C語言(3288) C++語言(3117) Java相關(2746) 疑難問題(2699) 單片機工控(2479) Web開發(1951) 網絡通信(1793) 數據庫相關(1767) VB基礎類(1755) PHP(1727) 開發(1646) 系統維護與使用區(1617) .NETCore(1586) 基礎和管理(1579) JavaEE(1566) C++(1527) 專題技術討論區(1515) Windows客戶端使用(1484) HtmlCss(1466) ASP.NET(1428) Unity3D(1354) VCL組件開發及應用(1353) HTML(CSS)(1220) 其他技術討論專區(1200) WindowsServer(1192) .NET技术(1165) 交換及路由技術(1149) 語言基礎算法系統設計(1133) WindowsSDKAPI(1124) 界面(1088) JavaSE(1075) Qt(1074) VBA(1048) 新手樂園(1016) 其他開發語言(947) Go(907) HTML5(901) 新技術前沿(898) 硬件設計(872) 區塊鏈技術(860) 網絡編程(857) 非技術版(846) 一般軟件使用(839) 網絡協議與配置(835) Eclipse(790) Spark(750) 下載資源懸賞專區(743)

熱門瀏覽
  • CA和證書

    1、在 CentOS7 中使用 gpg 創建 RSA 非對稱密鑰對 gpg --gen-key #Centos上生成公鑰/密鑰對(存放在家目錄.gnupg/) 2、將 CentOS7 匯出的公鑰,拷貝到 CentOS8 中,在 CentOS8 中使用 CentOS7 的公鑰加密一個檔案 gpg -a ......

    uj5u.com 2020-09-10 00:09:53 more
  • Kubernetes K8S之資源控制器Job和CronJob詳解

    Kubernetes的資源控制器Job和CronJob詳解與示例 ......

    uj5u.com 2020-09-10 00:10:45 more
  • VMware下安裝CentOS

    VMware下安裝CentOS 一、軟硬體準備 1 Centos鏡像準備 1.1 CentOS鏡像下載地址 下載地址 1.2 CentOS鏡像下載程序 點擊下載地址進入如下圖的網站,選擇需要下載的版本,這里選擇的是Centos8,點擊如圖所示。 決定選擇Centos8后,選擇想要的鏡像源進行下載,此 ......

    uj5u.com 2020-09-10 00:12:10 more
  • 如何使用Grep命令查找多個字串

    如何使用Grep 命令查找多個字串 大家好,我是良許! 今天向大家介紹一個非常有用的技巧,那就是使用 grep 命令查找多個字串。 簡單介紹一下,grep 命令可以理解為是一個功能強大的命令列工具,可以用它在一個或多個輸入檔案中搜索與正則運算式相匹配的文本,然后再將每個匹配的文本用標準輸出的格式 ......

    uj5u.com 2020-09-10 00:12:28 more
  • git配置http代理

    git配置http代理 經常遇到克隆 github 慢的問題,這里記錄一下幾種配置 git 代理的方法,解決 clone github 過慢。 目錄 git配置代理 git單獨配置github代理 git配置全域代理 配置終端環境變數 git配置代理 主要使用 git config 命令 git單獨 ......

    uj5u.com 2020-09-10 00:12:33 more
  • Linux npm install 裝包時提示Error EACCES permission denied解

    npm install 裝包時提示Error EACCES permission denied解決辦法 ......

    uj5u.com 2020-09-10 00:12:53 more
  • Centos 7下安裝nginx,使用yum install nginx,提示沒有可用的軟體包

    Centos 7下安裝nginx,使用yum install nginx,提示沒有可用的軟體包。 18 (flaskApi) [[email protected] flaskDemo]# yum -y install nginx 19 已加載插件:fastestmirror, langpacks 20 Loading ......

    uj5u.com 2020-09-10 00:13:13 more
  • Linux查看服務器暴力破解ssh IP

    在公網的服務器上經常遇到別人爆破你服務器的22埠,用來挖礦或者干其他嘿嘿嘿的事情~ 這種情況下正確的做法是: 修改默認ssh的22埠 使用設定密鑰登錄或者白名單ip登錄 建議服務器密碼為復雜密碼 創建普通用戶登錄服務器(root權限過大) 建立堡壘機,實作統一管理服務器 統計爆破IP [root ......

    uj5u.com 2020-09-10 00:13:17 more
  • CentOS 7系統常見快捷鍵操作方式

    Linux系統中一些常見的快捷方式,可有效提高操作效率,在某些時刻也能避免操作失誤帶來的問題。 ......

    uj5u.com 2020-09-10 00:13:31 more
  • CentOS 7作業系統目錄結構介紹

    作業系統存在著大量的資料檔案資訊,相應檔案資訊會存在于系統相應目錄中,為了更好的管理資料資訊,會將系統進行一些目錄規劃,不同目錄存放不同的資源。 ......

    uj5u.com 2020-09-10 00:13:35 more
最新发布
  • Java不等待異步呼叫回應

    我ExecutorService在 Java 中使用創建一個執行緒池,其中每個執行緒執行一些異步任務/呼叫另一個服務。我不想等待回應,而是在回應到達時回傳回應。ExecutorService executor =...

    uj5u.com 2021-10-16 16:57:59 more
  • AsyncFunction().GetAwaiter().GetResult()和Task.Run(()=>Async

    我試圖理解GetAwaiter().GetResult(). 我知道如果可能的話不應該使用它,但在這種情況下我不能使用 async/await,所以我必須從同步問題中的異步函式中獲取結果。讓我們考慮這個...

    uj5u.com 2021-10-16 16:57:27 more
  • 如何按順序等待C#任務,而不是并行等待?

    我有一組在外部硬體上運行的異步測驗。我可以按順序運行它們,但因為所有這些測驗都有副作用,我希望能夠對它們進行洗牌并一遍又一遍地運行它們。當我將它們放在一個串列中并嘗...

    uj5u.com 2021-10-16 16:56:56 more
  • 如何使node.js功能異步?有時仍然回傳空物件

    我的 Node.js 路由 ( result) 中有一個物件,我想將其拆分為兩個單獨的物件并將它們都回傳。但是,有時第二個物件回傳空并修復我想讓拆分它的函式異步的問題。但由于某種原因它...

    uj5u.com 2021-10-16 16:56:19 more
  • 如何在F#中簡化異步編程

    我來自使用 async/await 的 C# 背景。我正在嘗試使用庫找到一種“不那么冗長”的編程方式。(特別是用于瀏覽器自動化的 Microsoft Playwright 庫)let (~~) = Async.AwaitTaskl...

    uj5u.com 2021-10-16 16:55:52 more
  • 為傳入的請求保持Websockets連接打開

    我有一個 Flask 服務器,它接受來自客戶端的 HTTP 請求。此 HTTP 服務器需要使用 websocket 連接將作業委托給第三方服務器(出于性能原因)。我發現很難理解如何創建一個可以為 H...

    uj5u.com 2021-10-16 16:55:23 more
  • 為什么每個檔案都成功加載后,我的網站會顯示加載圖示?

    所以我正在學習 JavaScript Promises 并創建了一個網頁,顯示從目錄中另一個檔案派生的文本。它成功加載了站點和所有內容,但問題是即使在站點加載和訪問其他網頁的資料后,它仍...

    uj5u.com 2021-10-16 16:54:46 more
  • 即使等待后沒有任何內容,是否會創建延續?

    我正在閱讀這篇文章并找到了這個例子:public static class DeadlockDemo{ private static async Task DelayAsync() { await Task.Delay(1000); } // This method cau...

    uj5u.com 2021-10-16 16:54:17 more
  • 如何渲染依賴API資料的React組件?

    我試圖在依賴于來自外部 API 的資料的組件檔案中呈現一個組件。基本上,我在組件中的回傳使用了一個正在等待資料的組件,但是我收到 dataRecords 未定義的錯誤,因此無法映射。希...

    uj5u.com 2021-10-16 16:53:42 more
  • 在另一個函式中使用異步函式的回傳資料

    我正在從 chess.com 資料中進行一些資料分析我有這個代碼const getUsernames = async function() { let response = await chessAPI.getCountryPlayers('RE') names = [.....

    uj5u.com 2021-10-16 16:53:09 more