无法将 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.ObservableCollection1.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);
    });
}