我有一个方法,我想在后台使用Task执行。然而,该方法需要引用对象作为参数。但是,该对象是在UI线程中创建的,因此我得到了“调用线程无法访问此对象,因为不同的线程拥有它”。例外。我应该怎么做呢?
Task.Factory.StartNew(() => SerializeGraphicsLayer(graphicsLayer, fileUrl))
.ContinueWith((t) => UpdateSaveOperation(t), TaskScheduler.FromCurrentSynchronizationContext());
SerializeGraphicsLayer()方法是我想在后台运行的方法,但是,我需要将在UI线程中创建的对象引用传递给此方法...
编辑: 该方法(在backgound中运行)可能会抛出异常,UpdateSaveOperation()方法将在UI中执行必要的错误消息。
我想有很好的异常处理,这就是为什么我选择Task来做它?
答案 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创建对象并授予调用者访问对象的权限。这样,后台任务可以安全地访问对象。