无法将COM对象转换为类类型

时间:2018-09-26 14:33:20

标签: c# exception uwp com runtimeexception

我正在制作图书馆管理软件。我已经将数据库管理分离到一个单独的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();
    }
}

2 个答案:

答案 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);
}