我正在咨询的公司有一个特定的业务要求,即某个WPF Windows的每个实例必须拥有自己的UI线程,并且在首次加载应用程序时不共享.NET Framework创建的默认UI线程。
从编码的角度来看,这很容易实现并且运行良好,直到在xaml中引入Telerik RadDocking控件。我直接从示例代码中复制并粘贴了xaml表单telerik的RadDocking示例,而不进行修改。当应用程序启动时,WindowWithTelerikDockingFromExample [貌似]的两个实例首先加载没有问题,事实上,窗口的第二个实例(标题为“单独的UI线程上的窗口......”)是可操作的,并且可以正常工作,“MainWindow”也是如此。直到您激活第二个窗口然后激活主窗口,然后切换回第二个窗口,引发以下异常:
“调用线程无法访问此对象,因为另一个线程拥有它。”
找到
的来源'c:\TB\117\WPF_Scrum\Release_WPF\Sources\Development\Controls\Docking\Docking\Parts\AutoHideArea.cs'. Checksum: MD5 {3e 1e cd 2a 97 89 30 7e c9 1c 28 c2 28 13 aa e9}
The file 'c:\TB\117\WPF_Scrum\Release_WPF\Sources\Development\Controls\Docking\Docking\Parts\AutoHideArea.cs' does not exist.
Looking in script documents for 'c:\TB\117\WPF_Scrum\Release_WPF\Sources\Development\Controls\Docking\Docking\Parts\AutoHideArea.cs'...
Looking in the projects for 'c:\TB\117\WPF_Scrum\Release_WPF\Sources\Development\Controls\Docking\Docking\Parts\AutoHideArea.cs'.
The file was not found in a project.
Looking in directory 'C:\Program Files\Microsoft Visual Studio 10.0\VC\crt\src\'...
Looking in directory 'C:\Program Files\Microsoft Visual Studio 10.0\VC\atlmfc\src\mfc\'...
Looking in directory 'C:\Program Files\Microsoft Visual Studio 10.0\VC\atlmfc\src\atl\'...
Looking in directory 'C:\Program Files\Microsoft Visual Studio 10.0\VC\atlmfc\include\'...
The debug source files settings for the active solution indicate that the debugger will not ask the user to find the file: c:\TB\117\WPF_Scrum\Release_WPF\Sources\Development\Controls\Docking\Docking\Parts\AutoHideArea.cs.
The debugger could not locate the source file 'c:\TB\117\WPF_Scrum\Release_WPF\Sources\Development\Controls\Docking\Docking\Parts\AutoHideArea.cs'.
这是我的代码:
App.xaml.cs:
public partial class App : Application
{
protected override void OnStartup(StartupEventArgs e)
{
this.ShutdownMode = System.Windows.ShutdownMode.OnLastWindowClose;
// Init the application's main window...
var mainWindow = new WindowWithTelerikDockingFromExample();
mainWindow.Title = "Main Window";
this.MainWindow = mainWindow;
mainWindow.Show();
// init another instance of the window with the telerik docking, on a seperate UI thread...
var thread = new Thread(() =>
{
SynchronizationContext.SetSynchronizationContext(new DispatcherSynchronizationContext(Dispatcher.CurrentDispatcher));
var window2 = new WindowWithTelerikDockingFromExample();
window2.Title = "Window on seperate UI Thread...";
window2.Show();
System.Windows.Threading.Dispatcher.Run();
window2.Closed += (s2, e2) =>
{
window2.Dispatcher.InvokeShutdown();
};
});
thread.SetApartmentState(ApartmentState.STA);
thread.Start();
base.OnStartup(e);
}
}
WindowWithTelerikDockingFromExample.xaml:
<Window x:Class="TelerikDockingThreadIssueExample.WindowWithTelerikDockingFromExample"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:telerik="http://schemas.telerik.com/2008/xaml/presentation"
Title="Window with xaml copy and pasted from Telerik example" Height="300" Width="300">
<Grid>
<telerik:RadDocking BorderThickness="0" Padding="0">
<telerik:RadDocking.DocumentHost>
<telerik:RadSplitContainer>
<telerik:RadPaneGroup>
<telerik:RadDocumentPane Header="Document 1" Title="Document 1" />
</telerik:RadPaneGroup>
</telerik:RadSplitContainer>
</telerik:RadDocking.DocumentHost>
<telerik:RadSplitContainer InitialPosition="DockedLeft">
<telerik:RadPaneGroup>
<telerik:RadPane Header="Pane Left 1" IsPinned="False">
<TextBlock Text="Pane Left 1" />
</telerik:RadPane>
<telerik:RadPane Header="Pane Left 2" IsPinned="False">
<TextBlock Text="Pane Left 2" />
</telerik:RadPane>
<telerik:RadPane Header="Pane Left 3" IsPinned="False">
<TextBlock Text="Pane Left 3" />
</telerik:RadPane>
<telerik:RadPane Header="Pane Left 4" IsPinned="False">
<TextBlock Text="Pane Left 4" />
</telerik:RadPane>
</telerik:RadPaneGroup>
</telerik:RadSplitContainer>
<telerik:RadSplitContainer InitialPosition="DockedRight">
<telerik:RadPaneGroup>
<telerik:RadPane Header="Pane Right 1" IsPinned="False">
<TextBlock Text="Pane Right 1" />
</telerik:RadPane>
</telerik:RadPaneGroup>
</telerik:RadSplitContainer>
<telerik:RadSplitContainer InitialPosition="DockedBottom">
<telerik:RadPaneGroup>
<telerik:RadPane Header="Pane Bottom 1" IsPinned="False">
<TextBlock Text="Pane Bottom 1" />
</telerik:RadPane>
</telerik:RadPaneGroup>
</telerik:RadSplitContainer>
</telerik:RadDocking>
</Grid>
</Window>
有什么想法吗?
答案 0 :(得分:1)
由于该公司是Telerik的付费客户,我可以使用他们的帐户通过支持票据向Telerik发送信息。以下是我发送给他们的摘录。它有点类似于我最初在StackOverflow上发布的问题,但是,我更进了一步并反编译了他们的RadDocking DLL并提出了他们需要做些什么来解决这个问题的建议。我很高兴地报告他们回答了这个问题:
我们有一个特定的业务要求,WPF Windows的特定实例必须各自拥有自己的UI线程,而不是使用在加载应用程序时创建的默认UI线程。从我们的经验来看,从编码的角度来看,相对容易实现,直到我们将Telerik RadDocking控件引入混合。 我附上了一个包含简单VS解决方案和两个主文件的zip文件。其中一个是app.xaml.cs,另一个是WindowWithTelerikDockingFromExample.xaml。窗口中的xaml直接从WPF Docking示例中复制并粘贴,没有任何更改。 app.xaml.cs文件有一个方法:OnStartup。 OnStartup方法做了两件事: 1.它首先创建一个Window实例,并将应用程序MainWindow设置为该实例。 2.然后它创建另一个Window实例,但在另一个线程下。 Window发布的两个实例没有问题且可以运行。直到用户激活其中一个,然后激活另一个,然后在抛出以下异常时再次激活另一个: “调用线程无法访问此对象,因为其他线程拥有它。”
我们认为问题出在Telerik.Windows.Controls.Docking.dll的AutoHideArea类中。我们自己使用您的免费Telerik JustDecompile工具来反编译Telerik.Windows.Controls.Docking.dll,以帮助调查问题可能发生的位置。 AutoHideArea中的“OnLoaded”方法连接“OnApplicationDeactivated”事件,如下所示:
private void OnLoaded(object sender, RoutedEventArgs e)
{
this.isLoaded = true;
base.SelectedIndex = -1;
if (!BrowserInteropHelper.IsBrowserHosted)
{
WeakReference weakReference = new WeakReference(this);
Window window = Window.GetWindow(this);
if (window != null)
{
window.SizeChanged += new SizeChangedEventHandler((objects, SizeChangedEventArgs a) => AutoHideArea.OnWindowEventOccured(weakReference));
window.LocationChanged += new EventHandler((object s, EventArgs a) => AutoHideArea.OnWindowEventOccured(weakReference));
}
if (Application.Current != null)
{
Application.Current.Deactivated += new EventHandler((object s, EventArgs a) => AutoHideArea.OnApplicationDeactivated(weakReference));
}
}
}
如果您遵循OnApplicationDeactivated方法,您会注意到它调用“area.CloseImmediately()”方法,如下所示:
private static void OnApplicationDeactivated(WeakReference target)
{
AutoHideArea autoHideArea;
if (target.IsAlive)
{
autoHideArea = target.Target as AutoHideArea;
}
else
{
autoHideArea = null;
}
AutoHideArea area = autoHideArea;
if (area != null)
{
area.CloseImmediately();
}
}
如果您一直遵循该方法,您会注意到它设置了base.SelectedIndex = -1,如下所示:
private void CloseImmediately()
{
this.OnLayoutChangeStarted();
base.SelectedIndex = -1;
this.OnLayoutChangeEnded();
}
我们建议将方法更改为如下所示:
private void CloseImmediately()
{
if (!System.Windows.Threading.Dispatcher.CurrentDispatcher.CheckAccess())
{
System.Windows.Threading.Dispatcher.CurrentDispatcher.Invoke(new Action(CloseImmediately),null);
return;
}
this.OnLayoutChangeStarted();
base.SelectedIndex = -1;
this.OnLayoutChangeEnded();
}
以下是Telerik的回复:
感谢您在RadDocking中报告此异常。该 控件未在多线程环境中测试,我们不知道 这些问题。这个问题记录在我们的PITS中 - http://www.telerik.com/support/pits.aspx#/public/wpf/15920。我们会 查看我们即将发布的版本的例外情况,我们将考虑 你对修复的建议。
感谢您的反馈。我很高兴更新您的Telerik积分 同样。
此致
乔治 Telerik
TRY TELERIK的最新产品 - WPF的EQATEC应用分析。 了解用户在您的应用程序中使用(或不使用)的功能。 了解您的受众。更好地定位它。明智地发展。