5

当我无权访问用于检索数据的客户端库时,我试图了解异步获取一组数据的正确代码。我指定了一个端点和一个日期范围,我应该检索一个播放列表列表。在 Start() 调用之后,我现在再也没有回来过。注意:这是在 WinForm 中运行的。我试图更好地理解任务,而不只是想跳到等待或 BackgroundWorker。我知道我在某个地方迷路了。

    private void GoButtonClick(object sender, EventArgs e)
    {
        string baseUrl = "http://someserver/api";
        var startDateTime = this._startDateTimePicker.Value;
        var endDateTime = this._endDateTimePicker.Value;
        _getPlaylistsFunc = delegate()
            {
                var client = new PlaylistExportClient(baseUrl);
                return client.GetPlaylistsByDateRange(startDateTime, endDateTime).ToList();
            };
        var task = new Task<List<Playlist>>(_getPlaylistsFunc);
        task.ContinueWith((t) => DisplayPlaylists(t.Result));
        task.Start();
    }

    private void DisplayPlaylists(List<Playlist> playlists)
    {
        _queueDataGridView.DataSource = playlists;
    }

更新 我做了这些更改,但现在应用程序似乎挂起 UI 线程。

    private void GoButtonClick(object sender, EventArgs e)
    {
        string baseUrl = "http://someserver/api";
        var startDateTime = this._startDateTimePicker.Value;
        var endDateTime = this._endDateTimePicker.Value;
        var token = Task.Factory.CancellationToken;

        var context = TaskScheduler.FromCurrentSynchronizationContext();
        Task.Factory.StartNew(() =>
            {
                var client = new PlaylistExportClient(baseUrl);
                _queueDataGridView.DataSource = client.GetPlaylistsByDateRange(startDateTime, endDateTime).ToList();

            },token,TaskCreationOptions.None,context);

    }
4

2 回答 2

3

我建议您使用基于任务的异步模式。这很简单:

private async void GoButtonClick(object sender, EventArgs e)
{
    string baseUrl = "http://someserver/api";
    var startDateTime = this._startDateTimePicker.Value;
    var endDateTime = this._endDateTimePicker.Value;
    var playlists = await Task.Run(() =>
    {
        var client = new PlaylistExportClient(baseUrl);
        return client.GetPlaylistsByDateRange(startDateTime, endDateTime).ToList();
    });
    _queueDataGridView.DataSource = playlists;
}

请注意,这将阻塞线程池线程;如果您可以将库修改为具有GetPlaylistsByDateRangeAsync方法,则可以提高效率。

编辑:如果您卡在 .NET 4.0 上,您可以安装Microsoft.Bcl.Async以获得完整的async/await功能。如果 - 由于某些莫名其妙的原因 - 你仍然不能使用async/ await,那么你可以这样做:

private void GoButtonClick(object sender, EventArgs e)
{
    string baseUrl = "http://someserver/api";
    var startDateTime = this._startDateTimePicker.Value;
    var endDateTime = this._endDateTimePicker.Value;
    var context = TaskScheduler.FromCurrentSynchronizationContext();
    Task.Run(() =>
    {
        var client = new PlaylistExportClient(baseUrl);
        return client.GetPlaylistsByDateRange(startDateTime, endDateTime).ToList();
    }).ContinueWith(t =>
    {
        _queueDataGridView.DataSource = t.Result;
    }, context);
}

但是,请注意,使用这种方法您的错误处理更加复杂。

于 2012-12-20T20:31:43.410 回答
2

看起来您正在分配给后台线程中的 UI 控件的属性。这通常是坏消息。当您这样做时,WPF 通常会引发异常,不确定 WinForms。

在后台线程中捕获数据,但在将其分配给 UI 控件之前切换回主 UI 线程。尝试使用类似的方式将数据发布到 UI 线程

    var uiSync = SynchronizationContext.Current;
    Task.Factory.StartNew(() =>
        {
            var client = new PlaylistExportClient(baseUrl);
            var list = client.GetPlaylistsByDateRange(...).ToList();
            uiSync.Post(() => _queueDataGridView.DataSource = list, null);
        },token,TaskCreationOptions.None,context);
于 2012-12-20T20:16:40.280 回答