【问题标题】:Can not filling the datagrid in new Task无法在新任务中填充数据网格
【发布时间】:2013-03-09 20:43:59
【问题描述】:

我有带有 DataGrid 的 WPF 项目。 我使用 MVVM 模式。 这是我的虚拟机的一部分:

class LibraryViewModel
{
    #region Members
    //private SimpleLibDBEntities _database;
    ObservableCollection<BooksViewModel> _books = new ObservableCollection<BooksViewModel>();
    int count = 0;
    int sizeOfdb = 1000000;
    #endregion

    public ObservableCollection<BooksViewModel> Books
    {
        get
        {
            return _books;
        }
        set
        {
            _books = value;
        }
    }
    public LibraryViewModel()
    {
        Task task = Task.Factory.StartNew(Generator);
    }
    private void Generator()
    {
        for (count = 0; count < sizeOfdb; count++)
        {
            _books.Add(new BooksViewModel { Book = new BooksSet { Id = count, Title = "Title"+count, Author = "Author", Publisher = "Publisher", Year = 1000, Note = "Note" } });
        }
    }

它有效,但我的 DataGrid 只显示大约 50 000 -100 000 个元素(随机)而不是我的 int sizeOfdb = 1000000 元素。为什么会这样?如何修复它?(没有“任务”一切正常)

在这个例子中我可以如何使用 async/await? 类似的东西? (不工作。尝试使用 Dispathcher?)

public LibraryViewModel()
{
    GeneratorAsync();
}
private async void GeneratorAsync()
{
        await Task.Factory.StartNew(()=>{
            for (count = 0; count < sizeOfdb; count++)
            {
                _books.Add(...);
            }
        });
    }

【问题讨论】:

  • 您真的不应该在 UI 中显示 50 000 个项目,更不用说 1 000 000 个了。

标签: c# mvvm datagrid async-await fill


【解决方案1】:

这里的基本问题是ObservableCollection 不是完全线程安全的。更具体地说,从一个线程修改集合同时从另一个线程读取可能不安全,ObservableCollection 在修改集合的线程上引发其CollectionChanged 事件。

我猜在这种情况下发生的事情是,您的后台任务会创建许多元素并将它们添加到集合中,然后再创建 UI 并进行绑定。此时,集合中的元素将添加到 DataGrid。但是,当添加下一项并且在线程池线程上引发 CollectionChanged 事件时,DataGrid 的事件处理程序现在在线程池线程上运行,这违反了它的线程亲和性。这会导致终止您的任务的异常(我猜 - 如果您在调试器下运行并打开用户未处理异常的中断,您应该会看到这一点)。

如果您预计生成集合项会很昂贵(例如,在真实版本中涉及数据库访问),您应该在后台线程上执行此操作,然后将它们添加到 UI 线程上的ObservableCollection - 但this question has a good discussion of this .

您可以看到,await/async 并没有真正帮助您尝试制定,因为您只是试图以即发即弃的“async void”开始包装一个即发即弃的任务。如果您只是尝试将创建移出 UI 线程,那么如何:

private async void AddBooks()
{
    var books = await Task.Run(() => GetBooks());
    foreach (var book in books)
    {
        _books.Add(book);
    }
}

private List<BooksViewModel> GetBooks()
{
    List<BooksViewModel> books = new List<BooksViewModel>();
    for (count = 0; count < sizeOfdb; count++)
    {
        books.Add(new BooksViewModel { Book = new BooksSet { Id = count, Title = "Title" + count, Author = "Author", Publisher = "Publisher", Year = 1000, Note = "Note" } });
    }
    return books;
}

在这种情况下,创建是在线程池线程上运行的,然后等待将您带回 UI 线程。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2012-03-13
    • 2011-07-16
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-08-08
    • 2013-12-19
    相关资源
    最近更新 更多