我正在制作图书馆管理软件。我已经将数据库管理分离到一个单独的c#库“ DataAccessLibrary”中。它包含DataAccess.cs:
public static event MyDelegate Event_AddBook;
public static void InitializeDatabase()
{
using (SqliteConnection db = new SqliteConnection("Filename=MyBooks.db"))
{
db.Open();
//Not necessary
}
}
public static void AddBook(Book book)
{
Event_AddBook.Invoke(book);
using (SqliteConnection db = new SqliteConnection("Filename=MyBooks.db"))
{
db.Open();
//Not necessary
}
}
public static ObservableCollection<Book> GetBooks()
{
ObservableCollection<Book> books = new ObservableCollection<Book>();
using (SqliteConnection db = new SqliteConnection("Filename=MyBooks.db"))
{
db.Open();
//Not necessary
}
return books;
}
}
将书籍添加到数据库后,它将调用该事件,以便通知我的YourBooks_View类(显示书籍)已添加书籍。
(注意:这本书是使用新窗口添加的。)
在其他文件中,我将一本书添加到数据库中,如下所示:
Book b = new Book
{
Title = title,
Author = author,
Publisher = publisher,
ISBN = isbn,
Quantity = quantity.GetValueOrDefault(),
CoverImageLocation = CoverImageUri
};
DataAccess.AddBook(b);
YourBooks类已订阅事件“ Event_AddBook”:
public sealed partial class YourBooks_View : Page
{
private BooksViewModel ViewModel { get; set; } = new BooksViewModel();
public YourBooks_View()
{
this.InitializeComponent();
DataAccess.Event_AddBook += new MyDelegate(this.GridView_AddBook);
}
private void GridView_AddBook(Book book)
{
ViewModel.Books.Add(book);
}
}
代码编译没有错误,但是在运行时,当我添加书时,此异常抛出在YourBooks_View.GridView_AddBook()行:
System.InvalidCastException HResult = 0x80004002 Message =无法将类型为“ System.Collections.Specialized.NotifyCollectionChangedEventHandler”的COM对象转换为类类型“ System.Collections.Specialized.NotifyCollectionChangedEventHandler”。表示COM组件的类型的实例不能转换为不表示COM组件的类型。但是,只要基础COM组件支持对接口IID的QueryInterface调用,就可以将它们强制转换为接口。 来源= System.Private.CoreLib 堆栈跟踪: 在System.StubHelpers.StubHelpers.GetCOMIPFromRCW_WinRTDelegate(Object objSrc,IntPtr pCPCMD,IntPtr&ppTarget) 在System.Collections.Specialized.NotifyCollectionChangedEventHandler.Invoke(对象发送者,NotifyCollectionChangedEventArgs e) 在System.Collections.ObjectModel.ObservableCollection
1.OnCollectionChanged(NotifyCollectionChangedEventArgs e) at System.Collections.ObjectModel.ObservableCollection
1.InsertItem(Int32索引,T项目)处 在System.Collections.ObjectModel.Collection`1.Add(T项目) 在C:\ Users \ Hemil \ source \ repos \ Bookshelf \ Bookshelf \ YourBooks_View.xaml.cs:line 54中的Bookshelf.YourBooks_View.GridView_AddBook(Book book)中 在C:\ Users \ Hemil \ source \ repos \ Bookshelf \ DataAccessLibrary \ DataAccess.cs:line 44中的DataAccessLibrary.DataAccess.AddBook(Book book)中 在C:\ Users \ Hemil \ source \ repos \ Bookshelf \ Bookshelf \ NewBook_View.xaml.cs:line 137中的Bookshelf.NewBook_View.d__3.MoveNext()中
我不知道这意味着什么。有没有办法解决这个问题,或者有没有办法以其他方式通知视图。
我的C#不太好。原谅我有任何错误或愚蠢的问题。
编辑:我正在使用Microsoft.Data.Sqlite容器中提供的Sqlite数据库。
编辑:我遵循了本教程:https://docs.microsoft.com/en-us/windows/uwp/data-access/sqlite-databases。我的目标是最新的Windows 10,因此我的应用程序未包含Sqlite
编辑:我将整个源代码上传到了gitlab上的公共存储库中:https://gitlab.com/rahem027/bookshelf2
编辑:我想我也应该包括BookViewModel源代码。好吧,这里是:
public class BooksViewModel
{
private ObservableCollection<Book> books { get; set; }
public ObservableCollection<Book> Books
{
get { return this.books; }
set
{
this.books = value;
}
}
public BooksViewModel()
{
books = DataAccess.GetBooks();
}
}
答案 0 :(得分:3)
出现此问题的原因是,您从新视图的线程更新了主视图的ViewModel
。当您单击Add
按钮时,触发了AddBook
事件,该事件在新视图的线程上,但是您在ViewModel
类中更新了主视图的YourBooks_View
。您应该使用CoreDispatcher.RunAsync方法为新视图安排UI线程上的工作。
因此,只需使用CoreDispatcher.RunAsync方法更改GridView_AddBook
方法的代码,以安排UI线程上的工作以添加新的Book
对象。
这是GridView_AddBook
方法的修改代码:
private async void GridView_AddBook(Book book)
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
ViewModel.Books.Add(book);
});
}
答案 1 :(得分:0)
您尝试添加到集合中的书本对象可能已附加到数据库中,当您对数据库执行操作时,请确保处置其上下文(取决于您所使用的数据库提供者使用)。
要解决此问题,您可以尝试将book对象复制到新实例,然后将其放入ViewModel Collection中,如下所示:
private void GridView_AddBook(Book book)
{
Book b = new Book
{
Title = book.title,
Author = book.author,
Publisher = book.publisher,
ISBN = book.isbn,
Quantity = book.quantity.GetValueOrDefault(),
CoverImageLocation = book.CoverImageUri
};
ViewModel.Books.Add(b);
}