本文记录一个 WPF 的已知问题,在通过 CollectionViewSource 获取到 CollectionView 之后,如果 CollectionViewSource 对象已被 GC 回收,将可能在调用 CollectionView 的 SortDescriptions 属性进行清空或者移除项时,也就是使用 SortDescriptionCollection 类型的清空或者移除项时,在 WPF 框架里面抛出空异常

此问题已经报告给 WPF 官方,请看 https://github.com/dotnet/wpf/issues/7389

我现在是一个成熟的开发者了,自己报告的 BUG 就要自己修。此问题已修复,请看 https://github.com/dotnet/wpf/pull/7390

此问题的复现步骤如下

在一个 WPF 项目里面,构建出一个 CollectionViewSource 对象,接着只获取存放此 CollectionViewSource 对象的 View 属性,此 View 属性就是 CollectionView 类型的一个对象,将 CollectionView 存放到字段里面。等待 CollectionViewSource 被回收之后,调用 CollectionView 的 SortDescriptions 属性进行清空 SortDescriptionCollection 的内容。代码如下

    public MainWindow()
    {
        InitializeComponent();

        var collectionViewSource = new CollectionViewSource()
        {
            Source = new List<Foo>(),
            IsLiveSortingRequested = true,
        };

        var collectionView = collectionViewSource.View;
        _collectionView = collectionView;

        collectionView.SortDescriptions.Add(new SortDescription("Name", ListSortDirection.Descending));

        Loaded += MainWindow_Loaded;
    }

    private readonly ICollectionView _collectionView;

    private void MainWindow_Loaded(object sender, RoutedEventArgs e)
    {
        GC.Collect();
        GC.WaitForFullGCComplete();
        GC.Collect();
    }

    private void Button_OnClick(object sender, RoutedEventArgs e)
    {
        _collectionView.SortDescriptions.Clear();
    }

以上的代码放在 githubgitee 欢迎访问

可以通过如下方式获取本文的源代码,先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin c7556d7b92605000011425f82793f9e4063e5a00

以上使用的是 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin c7556d7b92605000011425f82793f9e4063e5a00

获取代码之后,进入 LechelaneHenayfucee 文件夹

运行代码,然后点击按钮,就可以看到在 WPF 框架里面抛出空异常

异常的调用堆栈大概如下

>	PresentationFramework.dll!System.Windows.Data.ListCollectionView.PrepareLocalArray() 
 	PresentationFramework.dll!System.Windows.Data.ListCollectionView.RefreshOverride() 
 	PresentationFramework.dll!System.Windows.Data.CollectionView.RefreshInternal() 
 	PresentationFramework.dll!System.Windows.Data.CollectionView.RefreshOrDefer() 
 	PresentationFramework.dll!System.Windows.Data.ListCollectionView.SortDescriptionsChanged(object sender, System.Collections.Specialized.NotifyCollectionChangedEventArgs e) 
 	WindowsBase.dll!System.ComponentModel.SortDescriptionCollection.RemoveItem(int index)
 	System.Private.CoreLib.dll!System.Collections.ObjectModel.Collection<System.ComponentModel.SortDescription>.RemoveAt(int index)
 	App.dll!MyClass.Foo();

阅读 WPF 框架的源代码,可以了解到原因就是因为 CollectionViewSource 对象没有被引用,从而被 GC 回收。在 CollectionViewSource 回收之后,将会让其 View 属性,也就是 CollectionView 类型,被 WPF 框架触发 DetachFromSourceCollection 方法进行回收。这个 DetachFromSourceCollection 方法代码如下

 public virtual void DetachFromSourceCollection() 
 { 
     INotifyCollectionChanged incc = _sourceCollection as INotifyCollectionChanged; 
     if (incc != null) 
     { 
         IBindingList ibl; 
         if (!(this is BindingListCollectionView) || 
             ((ibl = _sourceCollection as IBindingList) != null && !ibl.SupportsChangeNotification)) 
         { 
             incc.CollectionChanged -= new NotifyCollectionChangedEventHandler(OnCollectionChanged); 
         } 
     } 
  
     _sourceCollection = null; 
 }

在 DetachFromSourceCollection 方法里面,将 _sourceCollection 设置为空,这就导致了在清空 SortDescriptionCollection 内容的时候,尝试获取 _sourceCollection 的属性时,抛出空异常


本文会经常更新,请阅读原文: https://blog.lindexi.com/post/WPF-%E5%B7%B2%E7%9F%A5%E9%97%AE%E9%A2%98-%E6%B8%85%E7%A9%BA-CollectionView-%E7%9A%84-SortDescriptions-%E5%8F%AF%E8%83%BD%E6%8A%9B%E5%87%BA%E7%A9%BA%E5%BC%82%E5%B8%B8.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

如果你想持续阅读我的最新博客,请点击 RSS 订阅,推荐使用RSS Stalker订阅博客,或者前往 CSDN 关注我的主页

知识共享许可协议 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名林德熙(包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系

无盈利,不卖课,做纯粹的技术博客

以下是广告时间

推荐关注 Edi.Wang 的公众号

欢迎进入 Eleven 老师组建的 .NET 社区

以上广告全是友情推广,无盈利