Entity 4.1 Обновление существующей родительской сущности новыми дочерними сущностями - PullRequest
27 голосов
/ 01 ноября 2011

У меня есть приложение, в котором вы можете создать новый тип продукта и добавить к нему некоторые ингредиенты. Продукт и ингредиенты являются объектами, сохраненными в базе данных. Сущность продукта имеет коллекцию компонентов-ингредиентов.

(упрощенная версия)

public class Product
   Public Sub New()
     Me.Ingredients = New List(Of Ingredient)()
   End Sub

   Property Ingredients as ICollection(Of Ingredient)
end class

Когда я сохраняю продукт в первый раз, все идет хорошо: я просто добавляю его в контекст и вызываю SaveChanges.

myDataContext.Products.Add(product)
myDataContext.SaveChanges()

И продукт (родитель), и ингредиенты (дети) сохраняются и связаны друг с другом. Все хорошо.

Однако, когда я добавляю / удаляю ингредиент в существующий продукт, у меня возникают проблемы. Сначала я очищаю существующую коллекцию ингредиентов в сущности продукта, а затем снова добавляю обновленный список ингредиентов (я не использую ингредиенты повторно, чтобы добавить момент). Затем я изменяю состояние объекта продукта на измененное и вызываю savechanges. Однако при изменении состояния я получаю исключение « Объект с таким же ключом уже существует в ObjectStateManager ».

myDataContext.Entry(product).State = EntityState.Modified

После «некоторого» поиска я выяснил, что проблема в том, что все ингредиенты имеют первичный ключ 0 (поскольку они еще не добавлены), а когда вы изменяете состояние родительского объекта (продукта), все дочерние элементы сущности (ингредиенты) присоединяются к контексту с ключом 0, что вызывает проблему, поскольку ключи больше не являются уникальными.

Я искал решение, но не могу понять, как решить эту проблему. Я пытался добавить ингредиенты в контекст перед изменением состояния, но тогда связь между продуктом и ингредиентами отсутствует ... Как обновить существующую родительскую сущность новыми, еще не добавленными дочерними сущностями?

Я использую Entity Framework 4.1 и Code First.

Надеюсь, вы можете мне помочь!

Ответы [ 4 ]

36 голосов
/ 01 ноября 2011

Сначала я очищаю существующую коллекцию ингредиентов в сущности продукта, а затем снова добавляю обновленный список ингредиентов.

Что ж, это своего рода атака методом перебора для обновлениядетская коллекция.В EF нет никакого волшебства для обновления дочерних элементов - что означает: добавление новых дочерних элементов, удаление удаленных дочерних элементов, обновление существующих дочерних элементов - путем установки состояния родителя только на Modified.По сути, эта процедура вынуждает вас также удалять старых детей из базы данных и вставлять новых, например:

// product is the detached product with the detached new children collection
using (var context = new MyContext())
{
    var productInDb = context.Products.Include(p => p.Ingredients)
        .Single(p => p.Id == product.Id);

    // Update scalar/complex properties of parent
    context.Entry(productInDb).CurrentValues.SetValues(product);

    foreach (var ingredient in productInDb.Ingredients.ToList())
        context.Ingredients.Remove(ingredient);

    productInDb.Ingredients.Clear(); // not necessary probably

    foreach (var ingredient in product.Ingredients)
        productInDb.Ingredients.Add(ingredient);

    context.SaveChanges();
}

Лучшая процедура - обновить коллекцию детей в памяти без удаления всех детей в базе данных.:

// product is the detached product with the detached new children collection
using (var context = new MyContext())
{
    var productInDb = context.Products.Include(p => p.Ingredients)
        .Single(p => p.Id == product.Id);

    // Update scalar/complex properties of parent
    context.Entry(productInDb).CurrentValues.SetValues(product);

    var ingredientsInDb = productInDb.Ingredients.ToList();
    foreach (var ingredientInDb in ingredientsInDb)
    {
        // Is the ingredient still there?
        var ingredient = product.Ingredients
            .SingleOrDefault(i => i.Id == ingredientInDb.Id);

        if (ingredient != null)
            // Yes: Update scalar/complex properties of child
            context.Entry(ingredientInDb).CurrentValues.SetValues(ingredient);
        else
            // No: Delete it
            context.Ingredients.Remove(ingredientInDb);
    }

    foreach (var ingredient in product.Ingredients)
    {
        // Is the child NOT in DB?
        if (!ingredientsInDb.Any(i => i.Id == ingredient.Id))
            // Yes: Add it as a new child
            productInDb.Ingredients.Add(ingredient);
    }

    context.SaveChanges();
}
4 голосов
/ 25 марта 2013

Я нашел эту недавнюю статью в расширении GraphDiff для DbContext.

Очевидно, это универсальный, многократно используемый вариант решения Слаума .

Пример кода:

using (var context = new TestDbContext())
{
    // Update DBcompany and the collection the company and state that the company 'owns' the collection Contacts.
    context.UpdateGraph(company, map => map.OwnedCollection(p => p.Contacts));     
    context.SaveChanges();
}

на примечании стороны; Я вижу, что автор предложил команде EF использовать свой код в выпуске # 864 Обеспечить лучшую поддержку для работы с отключенными объектами .

0 голосов
/ 28 июня 2013

Я считаю, это более простое решение.

public Individual
{
.....

public List<Address> Addresses{get;set;}


}

//where base.Update from Generic Repository
public virtual void Update(T entity)
        {
            _dbset.Attach(entity);
            _dataContext.Entry(entity).State = EntityState.Modified;
        }

//overridden update
 public override void Update(Individual entity)
        {


            var entry = this.DataContext.Entry(entity);
            var key = Helper.GetPrimaryKey(entry);
            var dbEntry = this.DataContext.Set<Individual>().Find(key);

            if (entry.State == EntityState.Detached)
            {
                if (dbEntry != null)
                {
                    var attachedEntry = this.DataContext.Entry(dbEntry);
                    attachedEntry.CurrentValues.SetValues(entity);
                }
                else
                {
                    base.Update(entity);
                }
            }
            else
            {
                base.Update(entity);
            }
            if (entity.Addresses.Count > 0)
            {
                foreach (var address in entity.Addresses)
                {
                    if (address != null)
                    {
                        this.DataContext.Set<Address>().Attach(address);
                        DataContext.Entry(address).State = EntityState.Modified;
                    }
                }
            }
        }
0 голосов
/ 04 января 2013

после многих и многих месяцев, пытаясь понять всю эту дрянную Entity Framework, я надеюсь, что это может помочь кому-то и не пережить разочарование, которое я пережил.

public void SaveOrder(SaleOrder order)
        {
            using (var ctx = new CompanyContext())
            {
                foreach (var orderDetail in order.SaleOrderDetails)
                {
                    if(orderDetail.SaleOrderDetailId == default(int))
                    {
                        orderDetail.SaleOrderId = order.SaleOrderId;
                        ctx.SaleOrderDetails.Add(orderDetail);
                    }else
                    {
                        ctx.Entry(orderDetail).State = EntityState.Modified;
                    }
                }

                ctx.Entry(order).State = order.SaleOrderId == default(int) ? EntityState.Added : EntityState.Modified;
                ctx.SaveChanges();                

            }

        }
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...