Code First adding to collections? How to use Code First with repositories?

前端 未结 6 1325
执笔经年
执笔经年 2020-12-17 14:58

EDIT: This happen only on larger scale projects with repositories. Is there anybody using EF4 with CodeFirst approach and using repositories? Please advise me.

Hi. I

相关标签:
6条回答
  • 2020-12-17 15:38

    Try use this signature. I hope this worked.

    public class Author
    {
        public virtual int AuthorId { get; set; }
        public virtual string Name { get; set; }
    
        private ICollection<Book> _books;
    
        public virtual ICollection<Book> Books
        {
            get { return _books ?? (_books = new HashSet<Book>()); } // Try HashSet<N>
            set { _books = value; }
        }
    
        public void AddBook(Book book)
        {
            book.Author = this;
            Books.Add(book);
        }
    }
    
    0 讨论(0)
  • 2020-12-17 15:39

    The problem in this instance is that change tracking proxies are being generated because all the properties on your class are marked as virtual. By convention this triggers Change Tracking Proxy generation.

    So you can either remove virtual keyword from one or more properties or simply tell the context not to generate change tracking proxies as per Working with POCO Entities.

    var ctx = new MyDbContext();
    ctx.Configuration.ProxyCreationEnabled = false;
    
    0 讨论(0)
  • 2020-12-17 15:40

    I've got code first working, but the only (major) difference with my code is I initialize it as a list rather than a collection... so my code reads something like:

    public class Author
    {
        public virtual int AuthorId { get; set; }
        public virtual string Name { get; set; }
        public virtual ICollection<Book> Books { get; set; }
    
        public Author()
        {
            Books = new List<Book>();
        }
    }
    

    Also, I just add directly to the Books collection - there is no need to add the book to the collection AND add the author to the book because the entity framework should take care of that.

    If this doesn't work for you. Let me know.

    HTHs,
    Charles

    0 讨论(0)
  • 2020-12-17 15:42

    Removing the "virtual" keyword from the collection properties works around the problem, by preventing the Entity Framework from creating a change tracking proxy. However, this is not a solution for many people, because change tracking proxies can be really convenient and can help prevent issues when you forget to detect changes at the right places in your code.

    A better approach would be to modify your POCO classes, so that they instantiate the collection properties in their get accessor, rather than in the constructor. Here's the original Author POCO class, modified to allow change tracking proxy creation:

    public class Author
    {
        public virtual int AuthorId { get; set; }
        public virtual string Name { get; set; }
    
        private ICollection<Book> _books;
    
        public virtual ICollection<Book> Books
        {
            get { return _books ?? (_books = new Collection<Book>()); }
            set { _books = value; }
        }
    
        public void AddBook(Book book)
        {
            book.Author = this;
            Books.Add(book);
        }
    }
    

    In the above code the collection property is no longer automatic, but rather has a backing field. It's better if you leave the setter protected, preventing any code (other than the proxy) from subsequently modifying these properties. You will notice that the constructor was no longer necessary and was removed.

    0 讨论(0)
  • 2020-12-17 15:57

    virtual on all the properties was the issue here for me.. Don't get why it have a impact like that when I did scaled it up to my project but that's how it was.. Hope this helps anybody that got the same issue..

    0 讨论(0)
  • 2020-12-17 15:58

    The issue mentioned above by Dejan.S regarding the use of 'List' to initialize a collection still occurs in EF5 RTM and ASP.NET 4.5 RTM:

    List init issue

    BUT, if you remove the virtual keyword from the primary key (e.g. UserId in the above screencap), it works! So it seems all of your properties can be virtual, EXCEPT the primary key.

    0 讨论(0)
提交回复
热议问题