4

我有一个 WPF 数据网格,绑定到由数据库中的 linq-to-sql 填充的列表。绑定是双向的,允许用户更改每行中的值

        <wpf:DataGrid AutoGenerateColumns="False" 
              ItemsSource="{Binding MyList}" 
              SelectedItem="{Binding SelectedItem}" >

当显示大约 20000 行时,程序在列表初始化期间因内存不足异常而崩溃。即使行数较少,性能也会变得难以忍受。

我知道在初始化时,数据网格会遍历每一行以测量最大列宽和其他属性。无论它们是否在屏幕上,它显然都会对所有行执行此操作。

我尝试将数据网格绑定到myQuery.ToList()(允许通过单击列对数据网格进行排序)或直接绑定到 IQueryable。(排序不适用于此)

两者都产生相同的结果。仅具有 20000 个项目的 ToList() 不会导致大量内存消耗,仅当它绑定到 datagrid 时才会发生这种情况。

忽略数据网格中的 20000 行有多有用的问题(这些是当前要求;更改这些工作示例会有所帮助)。

什么是最简单的方法来延迟加载当前显示在屏幕上的数据,而忽略其他所有内容,直到它被滚动到视图中?

这可以在没有第三方库和主要代码更改的情况下完成吗?

如果没有,推荐的解决方法是什么?

4

2 回答 2

5

事实证明,问题完全是我的用户错误:

WPF Datagrid 可以很好地进行 UI 虚拟化:仅在需要时才绘制消耗内存的行对象;如果一行超出数据网格的可见边界,则不会实例化。

但是,如果数据网格包含在 ScrollViewer 中,这将不起作用。在滚动查看器中,数据网格的每个部分都是虚拟可见的,因此将呈现整个数据网格。然后,滚动查看器仅显示此呈现的数据网格中适合 UI 窗口的部分。

由于滚动查看器中的数据网格看起来就像管理自己的滚动条的数据网格,所以我没有注意到滚动查看器。

移除滚动查看器后,即使高度和宽度可变,大量行也不会造成任何问题。数据网格只是填充可用空间,并且只根据需要实例化新行。

所以简而言之,我的问题的解决方案是:不要将数据网格放在滚动查看器中

于 2012-06-05T16:27:11.953 回答
0

在这种情况下,您要绑定的属性假设您MyList由 MyFile 对象 ( List<MyFile>) 组成,那么您必须将 MyFile 类创建为:

 class MyFile
    {
        public string FullPath { get; set; }

        public string Extension
        {
            get
            {
                return Path.GetExtension(FullPath);
            }
        }

        public string PathRoot
        {
            get
            {
                return Path.GetPathRoot(FullPath);
            }
        }

        public DateTime CreationTime
        {
            get
            {
                return File.GetCreationTime(FullPath);
            }
        }

    }

这样,您将在每个对象中存储更少的信息,并在网格中显示的少数项目上调用 get 方法。与将实际值存储在类中相比。希望这可以帮助

于 2012-05-03T01:31:14.247 回答