5

我正在修改 Windows 窗体以允许在 UI 保持响应时在后台加载数据。数据检索和绑定都需要相当长的时间。理想情况下,我会在后台同时进行这两项操作,但是对于我应该在后台(如在主线程之外)进行何种 UI 更新存在一些歧义。一个在后台显示数据检索和数据绑定的可靠示例将非常有帮助。

4

3 回答 3

6

检索可以而且应该被推到后台线程——但是有一些模式可以将其全部到位。

基本上,您将启动一个后台线程来检索数据,一旦完成,它将需要合并回 UI 线程以执行实际的 UI 更新(跨线程的 UI 更新非常糟糕)。

三种基本的后台线程方式供您探索

  • 最简单/最有限(和古怪的 IMO)是 BackgroundWorker 组件
  • 使用 Delegates 及其 BeginInvoke()/EndInvoke() 方法可以很好地平衡易用性和灵活性(并使用 ThreadPool 线程)
  • 使用原始 Thread 对象提供了最多的控制,但设置起来比 ThreadPool 线程慢

我个人倾向于代表选项;一旦你掌握了模式,它们就很容易使用。BackgroundWorker 看起来不错,但有一些陷阱和缺失的管道,使其使用起来比您预期的更麻烦。让我整理一下代表方法的简短示例;我会尽快更新...

编辑

这是一些代码,它在 VB 中,但如果你是 C# 人,应该很容易转录。对于希望后台线程的行为方式,您还有更多选择,因此这里有两个示例。非阻塞是我的首选,但如果您将其安装到现有代码中,那么阻塞可能对您来说更容易。

非阻塞,一旦后台线程完成,将在UI线程上调用回调方法(GetData_Complete)

Sub Main()

    Console.WriteLine("On the main thread")
    Dim dataDelegate As New GetDataCaller(AddressOf GetData)

    Dim iar As IAsyncResult

    ' Non-blocking approach using a callback method
    iar = dataDelegate.BeginInvoke(AddressOf GetData_Complete, Nothing)

End Sub

Private Delegate Sub GetData_CompleteCaller(ByVal iar As IAsyncResult)
Private Sub GetData_Complete(ByVal iar As IAsyncResult)
    If InvokeRequired Then
        Dim invokeDelegate As New GetData_CompleteCaller(AddressOf GetData_Complete)
        Invoke(invokeDelegate, New Object() {iar})
        Exit Sub
    End If

    ' Downcast the IAsyncResult to an AsyncResult -- it's safe and provides extra methods
    Dim ar As System.Runtime.Remoting.Messaging.AsyncResult = DirectCast(iar, System.Runtime.Remoting.Messaging.AsyncResult)

    Dim dataDelegate As GetDataCaller = DirectCast(ar.AsyncDelegate, GetDataCaller)
    Dim result As String = dataDelegate.EndInvoke(iar)

    Console.WriteLine("On the main thread again, background result is: " + result)

End Sub

Private Delegate Function GetDataCaller() As String
Private Function GetData() As String
    Console.WriteLine("On the background thread!")

    For index As Integer = 0 To 2
        Console.WriteLine("Background thread is working")
    Next

    Return "Yay, background thread got the data!"

End Function

阻塞 子 Main()

    Console.WriteLine("On the main thread")
    Dim dataDelegate As New GetDataCaller(AddressOf GetData)

    Dim iar As IAsyncResult

    ' blocking approach; WaitOne() will block this thread from proceeding until the background thread is finished
    iar = dataDelegate.BeginInvoke(Nothing, Nothing)
    iar.AsyncWaitHandle.WaitOne()
    Dim result As String = dataDelegate.EndInvoke(iar)
    Console.WriteLine("On the main thread again, background result is: " + result)

End Sub

Private Sub GetData_Complete(ByVal iar As IAsyncResult)

    ' Downcast the IAsyncResult to an AsyncResult -- it's safe and provides extra methods
    Dim ar As System.Runtime.Remoting.Messaging.AsyncResult = DirectCast(iar, System.Runtime.Remoting.Messaging.AsyncResult)

    Dim dataDelegate As GetDataCaller = DirectCast(ar.AsyncDelegate, GetDataCaller)
    Dim result As String = dataDelegate.EndInvoke(iar)

    Console.WriteLine("On the main thread again, background result is: " + result)

End Sub

Private Delegate Function GetDataCaller() As String
Private Function GetData() As String
    Console.WriteLine("On the background thread!")

    For index As Integer = 0 To 2
        Console.WriteLine("Background thread is working")
    Next

    Return "Yay, background thread got the data!"

End Function
于 2009-07-10T15:12:57.797 回答
1

永远不要从任何后台线程更新 UI,一旦您从服务器调用回 UI 线程以更新 UI 绑定到的 UI 控件或数据集的数据。

在这种情况下,使用 BackgroundWorker 将有助于连接事件。

高温高压

菲尔

于 2009-07-10T15:08:20.800 回答
0

加载(如“从数据源检索”)可能是微不足道的,无论您使用委托、后台工作人员还是任何其他协议。但是绑定似乎很棘手,因为至少在大多数数据绑定控件中无法对其施加太多控制 - 您可以异步检索数据,但是一旦您准备好如何将其提供给后台的大型网格?那是你的问题吗?如果是这样,我认为您可以:

  • 创建(或子类化)您的视图控件,为异步加载提供接口;
  • 实现一个分页视图,一次只显示 N 条记录,这样在获取/格式化记录时 UI 不会被阻塞。
于 2009-07-10T17:03:45.613 回答