无法将 COM 对象转换为 class 类型
Unable to cast COM Object to class type
我在做图书馆管理软件。我已将数据库管理分离到一个单独的 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 class(显示书籍)收到一本书被添加的通知。
(注意:本书是使用新的 window.)
添加的
在其他一些文件中,我将一本书添加到数据库中,如下所示:
Book b = new Book
{
Title = title,
Author = author,
Publisher = publisher,
ISBN = isbn,
Quantity = quantity.GetValueOrDefault(),
CoverImageLocation = CoverImageUri
};
DataAccess.AddBook(b);
YourBooks class 订阅了事件 '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=Unable to cast COM object of type 'System.Collections.Specialized.NotifyCollectionChangedEventHandler' to class type 'System.Collections.Specialized.NotifyCollectionChangedEventHandler'. Instances of types that represent COM components cannot be cast to types that do not represent COM components; however they can be cast to interfaces as long as the underlying COM component supports QueryInterface calls for the IID of the interface.
Source=System.Private.CoreLib
StackTrace:
at System.StubHelpers.StubHelpers.GetCOMIPFromRCW_WinRTDelegate(Object objSrc, IntPtr pCPCMD, IntPtr& ppTarget)
at System.Collections.Specialized.NotifyCollectionChangedEventHandler.Invoke(Object sender, NotifyCollectionChangedEventArgs e)
at System.Collections.ObjectModel.ObservableCollection1.OnCollectionChanged(NotifyCollectionChangedEventArgs e)
at System.Collections.ObjectModel.ObservableCollection
1.InsertItem(Int32 index, T item)
at System.Collections.ObjectModel.Collection`1.Add(T item)
at Bookshelf.YourBooks_View.GridView_AddBook(Book book) in C:\Users\Hemil\source\repos\Bookshelf\Bookshelf\YourBooks_View.xaml.cs:line 54
at DataAccessLibrary.DataAccess.AddBook(Book book) in C:\Users\Hemil\source\repos\Bookshelf\DataAccessLibrary\DataAccess.cs:line 44
at Bookshelf.NewBook_View.d__3.MoveNext() in C:\Users\Hemil\source\repos\Bookshelf\Bookshelf\NewBook_View.xaml.cs:line 137
我不知道这是什么意思。有没有办法解决这个问题,或者有没有办法以其他方式通知视图。
我的c#不太好。请原谅我的任何错误或愚蠢的问题。
编辑: 我正在使用 Microsoft.Data.Sqlite 容器中提供的 Sqlite 数据库。
编辑: 我遵循了本教程:https://docs.microsoft.com/en-us/windows/uwp/data-access/sqlite-databases。我的目标是最新的 Windows 10,所以我没有在我的应用程序中包含 Sqlite
编辑: 我将整个源代码上传到 gitlab 上的 public 存储库: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();
}
}
您尝试添加到集合中的书籍对象可能附加到数据库,当您对数据库执行操作时,请确保 Dispose 它的上下文(这取决于您正在使用哪个数据库提供商)。
因此,要解决这个问题,您可以尝试将 book 对象复制到一个新实例并将其放入您的 ViewModel 集合中,如下所示:
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);
}
您遇到此问题的原因是您从新视图的线程更新了主视图的 ViewModel
。当您单击 Add
按钮时,触发了 AddBook
事件,这是在新视图的线程上,但是您在 YourBooks_View
[=27= 中更新了主视图的 ViewModel
].您应该使用 CoreDispatcher.RunAsync 方法为新视图安排 UI 线程上的工作。
因此只需将 GridView_AddBook
方法的代码更改为 CoreDispatcher.RunAsync 方法即可安排 UI 线程上的工作以添加新的 Book
对象。
这里是GridView_AddBook
方法的修改代码:
private async void GridView_AddBook(Book book)
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
ViewModel.Books.Add(book);
});
}
我在做图书馆管理软件。我已将数据库管理分离到一个单独的 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 class(显示书籍)收到一本书被添加的通知。
(注意:本书是使用新的 window.)
添加的在其他一些文件中,我将一本书添加到数据库中,如下所示:
Book b = new Book
{
Title = title,
Author = author,
Publisher = publisher,
ISBN = isbn,
Quantity = quantity.GetValueOrDefault(),
CoverImageLocation = CoverImageUri
};
DataAccess.AddBook(b);
YourBooks class 订阅了事件 '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=Unable to cast COM object of type 'System.Collections.Specialized.NotifyCollectionChangedEventHandler' to class type 'System.Collections.Specialized.NotifyCollectionChangedEventHandler'. Instances of types that represent COM components cannot be cast to types that do not represent COM components; however they can be cast to interfaces as long as the underlying COM component supports QueryInterface calls for the IID of the interface. Source=System.Private.CoreLib StackTrace: at System.StubHelpers.StubHelpers.GetCOMIPFromRCW_WinRTDelegate(Object objSrc, IntPtr pCPCMD, IntPtr& ppTarget) at System.Collections.Specialized.NotifyCollectionChangedEventHandler.Invoke(Object sender, NotifyCollectionChangedEventArgs e) at System.Collections.ObjectModel.ObservableCollection
1.OnCollectionChanged(NotifyCollectionChangedEventArgs e) at System.Collections.ObjectModel.ObservableCollection
1.InsertItem(Int32 index, T item) at System.Collections.ObjectModel.Collection`1.Add(T item) at Bookshelf.YourBooks_View.GridView_AddBook(Book book) in C:\Users\Hemil\source\repos\Bookshelf\Bookshelf\YourBooks_View.xaml.cs:line 54 at DataAccessLibrary.DataAccess.AddBook(Book book) in C:\Users\Hemil\source\repos\Bookshelf\DataAccessLibrary\DataAccess.cs:line 44 at Bookshelf.NewBook_View.d__3.MoveNext() in C:\Users\Hemil\source\repos\Bookshelf\Bookshelf\NewBook_View.xaml.cs:line 137
我不知道这是什么意思。有没有办法解决这个问题,或者有没有办法以其他方式通知视图。
我的c#不太好。请原谅我的任何错误或愚蠢的问题。
编辑: 我正在使用 Microsoft.Data.Sqlite 容器中提供的 Sqlite 数据库。
编辑: 我遵循了本教程:https://docs.microsoft.com/en-us/windows/uwp/data-access/sqlite-databases。我的目标是最新的 Windows 10,所以我没有在我的应用程序中包含 Sqlite
编辑: 我将整个源代码上传到 gitlab 上的 public 存储库: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();
}
}
您尝试添加到集合中的书籍对象可能附加到数据库,当您对数据库执行操作时,请确保 Dispose 它的上下文(这取决于您正在使用哪个数据库提供商)。
因此,要解决这个问题,您可以尝试将 book 对象复制到一个新实例并将其放入您的 ViewModel 集合中,如下所示:
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);
}
您遇到此问题的原因是您从新视图的线程更新了主视图的 ViewModel
。当您单击 Add
按钮时,触发了 AddBook
事件,这是在新视图的线程上,但是您在 YourBooks_View
[=27= 中更新了主视图的 ViewModel
].您应该使用 CoreDispatcher.RunAsync 方法为新视图安排 UI 线程上的工作。
因此只需将 GridView_AddBook
方法的代码更改为 CoreDispatcher.RunAsync 方法即可安排 UI 线程上的工作以添加新的 Book
对象。
这里是GridView_AddBook
方法的修改代码:
private async void GridView_AddBook(Book book)
{
await Dispatcher.RunAsync(CoreDispatcherPriority.Normal, () =>
{
ViewModel.Books.Add(book);
});
}