WPF 已知问题 清空 CollectionView 的 SortDescriptions 可能抛出空异常
此问题已经报告给 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();
}
可以通过如下方式获取本文的源代码,先创建一个空文件夹,接着使用命令行 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/
如图片看不见,请在浏览器开启不安全http内容兼容
本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。欢迎转载、使用、重新发布,但务必保留文章署名[林德熙](https://www.cnblogs.com/lindexi)(包含链接:https://www.cnblogs.com/lindexi ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请与我[联系](mailto:lindexi_gd@163.com)。