0

我有 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(...);
            }
        });
    }
4

1 回答 1

4

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

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

如果您希望生成集合项会很昂贵(例如,在真实版本中涉及数据库访问),您应该在后台线程上执行此操作,然后将它们添加到ObservableCollectionUI 线程上 - 但是这个问题对此有很好的讨论

您可以看到 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 线程。

于 2013-03-09T12:26:03.073 回答