Telerik RadDocking的线程问题

时间:2013-10-08 00:37:43

标签: wpf multithreading telerik dispatcher raddocking

我正在咨询的公司有一个特定的业务要求,即某个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>

有什么想法吗?

1 个答案:

答案 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应用分析。   了解用户在您的应用程序中使用(或不使用)的功能。   了解您的受众。更好地定位它。明智地发展。