如何将对象引用传递给Task(TPL)?

时间:2011-11-21 10:14:31

标签: c# wpf task-parallel-library

我有一个方法,我想在后台使用Task执行。然而,该方法需要引用对象作为参数。但是,该对象是在UI线程中创建的,因此我得到了“调用线程无法访问此对象,因为不同的线程拥有它”。例外。我应该怎么做呢?

Task.Factory.StartNew(() => SerializeGraphicsLayer(graphicsLayer, fileUrl))
                    .ContinueWith((t) => UpdateSaveOperation(t), TaskScheduler.FromCurrentSynchronizationContext()); 

SerializeGraphicsLayer()方法是我想在后台运行的方法,但是,我需要将在UI线程中创建的对象引用传递给此方法...

编辑: 该方法(在backgound中运行)可能会抛出异常,UpdateSaveOperation()方法将在UI中执行必要的错误消息。

我想有很好的异常处理,这就是为什么我选择Task来做它?

3 个答案:

答案 0 :(得分:1)

UI对象不适合在后台线程上使用。如果可能,您可以获得所需的,并捕获它们进入您的操作:

string name = userName.Text; // access values on UI thread
Task.Factory.StartNew(() => /* something using name, but not the UI control*/ );

但是,如果您需要将值更新更新到UI,则必须返回UI线程。如果你正在做的 all 正在更新UI,你也可以不使用TPL(这不太合适)。

答案 1 :(得分:1)

在WPF中,您可以使用Dispatcher类执行非UI相关的繁重任务,这些任务可以轻松移动到后台线程。有关更多说明,请查看以下文章:Understanding “Dispatcher” in WPF

通过Dispatcher调用方法时,可以将任何对象作为参数传递,并且在调用的方法中,您必须将其强制转换为原始类型。

答案 2 :(得分:0)

如果后台任务必须创建一个对象,该对象随后将由UI显示,则可以首先让UI调度程序创建该对象,然后使用Dispatcher 签名将其传递回后台任务。示例:

private Info AddFolderInfoRecursive(string folder)
{
        ....
        return Dispatcher.Invoke<Info>(() =>
        {
            Info info = new Info(folder)
            {
                Size = folderSize,
                NumFolders = numFolders,
                NumFiles = numFiles
            };
            return info;
        }
}

然后,调度程序调用UI,UI创建对象并授予调用者访问对象的权限。这样,后台任务可以安全地访问对象。