异步任务仍然阻止UI线程

时间:2016-07-20 08:57:02

标签: c# multithreading asynchronous

我正试着试探async/awaitTasks。我试图让一个方法失效,以便我可以在整个程序中使用它。基本上我有自己的BusyIndicator版本,我希望在工作完成时显示。这是一个基本的例子;

private async void OnPageLoad(object sender, RoutedEventArgs e)
{
    var waitWindow = new PleaseWaitWindow();

    waitWindow.Show();

    await LoadCompanyContracts();

    waitWindow.Close();
}

private async Task LoadCompanyContracts()
{
    await Task.Run(() =>
    {
        Dispatcher.Invoke(() =>
        {
             //Work done here
        });
    });     
}

即使通过这两种方法并尝试实施Tasks我的BusyIndicator仍然没有旋转,它会被显示但是被冻结,所以我相信UI线程仍然被阻止。如何修改这段代码以确保所有CPU绑定的工作都没有阻止UI线程?

3 个答案:

答案 0 :(得分:5)

作为一般规则,您根本不应使用Dispatcher。这是非常普遍的做法,但这是一种常见的不良做法。

private async Task LoadCompanyContracts()
{
  await Task.Run(() =>
  {
    //Work done here
  });
  // Update UI if necessary.
}

答案 1 :(得分:1)

不要在 Dispatcher.Invoke()中工作,因为您的调度程序与ui线程关联。在Task中工作并仅使用调度程序来更新UI

    private async Task LoadCompanyContracts()
    {
       await Task.Run(() =>
       {
         /*Work done here
          *
          */
         Dispatcher.Invoke(() =>
         {
             //Update UI 
         });
       });     
     }

答案 2 :(得分:-1)

在我的代码中查找一些示例后,您可以尝试使用该解决方案:

    private async void OnPageLoad(object sender, RoutedEventArgs e)
    {
        var waitWindow = new PleaseWaitWindow();

        waitWindow.Show();

        InitLoadCompanyContracts();

        waitWindow.Close();
    }


    private async void InitLoadCompanyContracts(){
        await LoadCompanyContracts();
    }


    private async Task LoadCompanyContracts()
    {
        await Task.Run(() =>
        {
            Dispatcher.Invoke(() =>
            {
                 //Work done here
            });
        });     
    }

我使用它来获得一个等待的方法,但不是所有的“图形”方法。我不知道是否是最好的方法,但对我而言,这是有效的。