【问题标题】:PictureBox animation freezing while Task running任务运行时PictureBox动画冻结
【发布时间】:2018-06-28 15:07:09
【问题描述】:

当长时间运行的操作运行时,我的 Winforms 应用程序在图片框内显示动画 gif。但是,它在等待任务完成时冻结:

Public Class MyUserControl
    Sub Initialize()
        Dim folderscantask = Task.Factory.StartNew(
            Function() EwsManagedApiScanFolderHierarchy(),
            TaskCreationOptions.LongRunning
            )
        folderdictask.Wait()
        Dim folderscanresult = folderscantask.Result
    End Sub

    Function EwsManagedApiScanFolderHierarchy() As Dictionary(Of String, String)
        'Performs a long, recursive operation involving a
        'Microsoft.Exchange.WebServices.Data.ExchangeService object
    End Function
End Class

为了保持 PictureBox 的动画运行,我应该做些什么不同的事情?

编辑

这是对我的问题的更完整描述,这次我使用了 Async/Await(因为我被告知 Task.Wait() 会阻塞调用者线程)。现在,动画移动正常,直到它第一次到达MyUserControl.BuildFolderMenus(),然后冻结。这是不可避免的吗?我的意思是,动画不是在专用线程中运行吗?

Public Class MyForm : Inherits Form

    'Form has a PictureBox named PictureBoxWaiting that shows an animated gif

    Public Async Sub MyButton_Click(sender as Object, e as EventArgs) Handles MyButton.Click
        PictureBoxWaiting.Show()
        PictureBoxWaiting.BringToFront()
        Await MyUserControl1.Initialize()
        PictureBoxWaiting.Hide()
        MyUserControl1.Show()
    End Sub

End Class

Public Class MyUserControl

    Public Async Function Initialize() As Task
        Dim folderdic = Await GetFolderHierarchyAsync()
        BuildFolderMenus(ToolStripDropDownButtonFolders, folderdic)
    End Function

    Public Async Function GetFolderHierarchyAsync() As Task(Of Dictionary(Of String, String))
        Return Await Task.Factory.StartNew(
            Function() EwsManagedApiScanFolderHierarchy(),
            TaskCreationOptions.LongRunning
            )
    End Function

    Function EwsManagedApiScanFolderHierarchy() As Dictionary(Of String, String)
        'Performs a long, recursive operation involving a
        'Microsoft.Exchange.WebServices.Data.ExchangeService object
    End Function

    Private Sub BuildFolderMenus(menu As ToolStripDropDownItem, dic As Dictionary(Of String, String))
        'This reads the dictionary containing the folder hierarchy
        'and recursively adds menu items in order that folders´
        'subfolders correspond to subitems inside an item
        '
        'This must run in UI thread since it creates UI controls
    End Sub

End Class

【问题讨论】:

  • 您对folderdictask.Wait() 的调用将阻塞当前线程,直到任务完成,因此您并没有完全异步运行它。查看async/await
  • Async/Await 是我的第一种方法,但是在扫描文件夹层次结构并将相应的菜单项添加到 ui 之前,它会使运行流过早地从该方法返回
  • 那么有些事情你做得不太正确,但如果你想在特定线路等待任务,你必须使用它。当然,在当前状态下,您也可以切换回旧的 ContinueWith() 方法。
  • 我认为你应该更关注为什么你的 async/await 方法不起作用,而不是尝试修复你当前的代码,因为 async/await 是一种执行异步工作的非常简单而强大的方法。也许递归是问题的一部分?
  • 不要为反对票出汗——我会说这是个人话题

标签: vb.net task


【解决方案1】:

您正在通过调用 Task.Wait() 阻塞 UI 线程。您需要使用 Asunc/Await 模式。例如创建一个这样的方法:

Public Async Function MyFunction() as Task
    Await Task.Run(Sub()
                        ' Do something non-UI which is time-consuming
                        ' This code runs in another thread without blocking UI
                        ' For example Thread.Sleep(5000)
                    End Sub)
    'The code here runs is UI thread
End Function

然后作为用法:

Private Async Sub Button1_Click(sender As Object, e As EventArgs) Handles Button1.Click
    Await MyUserControl1.MyFunction()
End Sub

然后你会看到,虽然你在MyFunction有一个耗时的任务,但是在任务运行的时候UI不会被阻塞。

【讨论】:

  • @VBobCat 的注意事项:要从异步函数返回某些内容,请使用返回类型 Task(Of yourType),在您的情况下为:Async Function EwsManagedApiScanFolderHierarchy() As Task(Of Dictionary(Of String, String))
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2020-08-10
  • 2018-11-29
  • 1970-01-01
  • 2012-04-21
相关资源
最近更新 更多