9

私はいくつかの重い仕事を実行するタスクを持っています。その結果をパスする必要がありますLogContent

Task<Tuple<SupportedComunicationFormats, List<Tuple<TimeSpan, string>>>>.Factory
    .StartNew(() => DoWork(dlg.FileName))
    .ContinueWith(obj => LogContent = obj.Result);

これはプロパティです:

public Tuple<SupportedComunicationFormats, List<Tuple<TimeSpan, string>>> LogContent
{
    get { return _logContent; }
    private set
    {
        _logContent = value;
        if (_logContent != null)
        {
            string entry = string.Format("Recognized {0} log file",_logContent.Item1);
            _traceEntryQueue.AddEntry(Origin.Internal, entry);
        }
    }
}

問題は、_traceEntryQueueデータが UI にバインドされていることです。そのため、このようなコードでは例外が発生します。

それで、私の質問はそれを正しく機能させる方法ですか?

4

4 に答える 4

16

ここに良い記事があります: Parallel Programming: Task Schedulers and Synchronization Context

Task.ContinueWith() メソッドを見てください。

例:

var context = TaskScheduler.FromCurrentSynchronizationContext();
var task = new Task<TResult>(() =>
    {
        TResult r = ...;
        return r;
    });

task.ContinueWith(t =>
    {
        // Update UI (and UI-related data) here: success status.
        // t.Result contains the result.
    },
    CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, context);

task.ContinueWith(t =>
    {
        AggregateException aggregateException = t.Exception;
        aggregateException.Handle(exception => true);
        // Update UI (and UI-related data) here: failed status.
        // t.Exception contains the occured exception.
    },
    CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, context);

task.Start();

.NET 4.5 はasync/awaitキーワードをサポートしているため ( Task.Run と Task.Factory.StartNewも参照):

try
{
    var result = await Task.Run(() => GetResult());
    // Update UI: success.
    // Use the result.
}
catch (Exception ex)
{
    // Update UI: fail.
    // Use the exception.
}
于 2013-01-31T11:14:22.800 に答える
5

UI スレッドで ContinueWith タスクを実行する必要があります。これは、 UI スレッドの TaskScheduler を、オーバーロードされたバージョンの ContinueWith メソッドと共に使用することで実現できます。

TaskScheduler scheduler = TaskScheduler.Current;
...ContinueWith(obj => LogContent = obj.Result), CancellationToken.None, TaskContinuationOptions.None, scheduler)
于 2013-01-31T07:06:58.897 に答える
1

を使用してDispatcher、UI スレッドでコードを呼び出すことができます。WPF ディスパッチャーの操作に関する記事をご覧ください。

于 2013-01-31T07:08:20.323 に答える
0

async/await を使用している場合、GUI スレッドで実行するタスクをスケジュールする方法を示すサンプル コードを次に示します。このコードをすべての async/await 呼び出しのスタックの一番下に配置して、GUI スレッドで実行されていないコードで WPF ランタイムがエラーをスローするのを回避します。

VS 2013 でテスト済みの WPF + MVVM で動作します。

public async Task GridLayoutSetFromXmlAsync(string gridLayoutAsXml)
{
    Task task = new Task(() => // Schedule some task here on the GUI thread );
    task.RunSynchronously();
    await task;
}
于 2014-09-09T15:49:17.353 に答える