Можно ли проверить, подключен ли объект к контексту данных в Entity Framework? - PullRequest
82 голосов
/ 11 ноября 2009

Я получаю следующую ошибку при попытке присоединить объект, который уже прикреплен к данному контексту через context.AttachTo(...):

Объект с таким же ключом уже существует в ObjectStateManager. ObjectStateManager не может отслеживать несколько объектов с одним и тем же ключом.

Есть ли способ достичь чего-то вроде:

context.IsAttachedTo(...)

Ура! * * 1013

Edit:

Метод расширения, описанный Джейсоном, близок, но он не работает для моей ситуации.

Я пытаюсь выполнить какую-то работу, используя метод, описанный в ответе на другой вопрос:

Как удалить одну или несколько строк из моей таблицы, используя Linq to Entities * без * предварительного извлечения строк?

Мой код выглядит примерно так:

var user = new User() { Id = 1 };
context.AttachTo("Users", user);
comment.User = user;
context.SaveChanges();

Это прекрасно работает, кроме случаев, когда я делаю что-то еще для этого пользователя, где я использую тот же метод и пытаюсь прикрепить фиктивный объект User. Это не удается, потому что я ранее прикрепил этот фиктивный объект пользователя. Как я могу проверить это?

Ответы [ 5 ]

56 голосов
/ 11 ноября 2009

Вот что у меня получилось, и это прекрасно работает:

public static void AttachToOrGet<T>(this ObjectContext context, string entitySetName, ref T entity)
    where T : IEntityWithKey
{
    ObjectStateEntry entry;
    // Track whether we need to perform an attach
    bool attach = false;
    if (
        context.ObjectStateManager.TryGetObjectStateEntry
            (
                context.CreateEntityKey(entitySetName, entity),
                out entry
            )
        )
    {
        // Re-attach if necessary
        attach = entry.State == EntityState.Detached;
        // Get the discovered entity to the ref
        entity = (T)entry.Entity;
    }
    else
    {
        // Attach for the first time
        attach = true;
    }
    if (attach)
        context.AttachTo(entitySetName, entity);
}

Вы можете назвать это следующим образом:

User user = new User() { Id = 1 };
II.AttachToOrGet<Users>("Users", ref user);

Это работает очень хорошо, потому что это похоже на context.AttachTo(...), за исключением того, что вы можете использовать трюк ID, который я цитировал выше каждый раз. Вы в конечном итоге либо с ранее прикрепленным объектом, либо со своим собственным объектом. Вызов CreateEntityKey для контекста гарантирует, что он хороший и общий и будет работать даже с составными ключами без дальнейшего кодирования (потому что EF уже может сделать это для нас!).

45 голосов
/ 31 октября 2013

Более простой подход:

 bool isDetached = context.Entry(user).State == EntityState.Detached;
 if (isDetached)
     context.Users.Attach(user);
18 голосов
/ 11 ноября 2009

Попробуйте этот метод расширения (это не проверено и не используется):

public static bool IsAttachedTo(this ObjectContext context, object entity) {
    if(entity == null) {
        throw new ArgumentNullException("entity");
    }
    ObjectStateEntry entry;
    if(context.ObjectStateManager.TryGetObjectStateEntry(entity, out entry)) {
        return (entry.State != EntityState.Detached);
    }
    return false;
}

Учитывая ситуацию, которую вы описываете при редактировании, вам может понадобиться следующая перегрузка, которая принимает EntityKey вместо объекта:

public static bool IsAttachedTo(this ObjectContext, EntityKey key) {
    if(key == null) {
        throw new ArgumentNullException("key");
    }
    ObjectStateEntry entry;
    if(context.ObjectStateManager.TryGetObjectStateEntry(key, out entry)) {
        return (entry.State != EntityState.Detached);
    }
    return false;
}

Чтобы построить EntityKey в вашей ситуации, используйте в качестве руководства следующее:

EntityKey key = new EntityKey("MyEntities.User", "Id", 1);

Вы можете получить EntityKey из существующего экземпляра User, используя свойство User.EntityKey (из интерфейса IEntityWithKey).

6 голосов
/ 11 ноября 2009

Используя ключ сущности объекта, который вы пытаетесь проверить:

var entry = context.ObjectStateManager.GetObjectStateEntry("EntityKey");
if (entry.State == EntityState.Detached)
{
  // Do Something
}

Доброжелательность,

Dan

0 голосов
/ 25 апреля 2017

Это не дает прямого ответа на вопрос ОП, но так я решил свой.

Это для тех, кто использует DbContext вместо ObjectContext.

    public TEntity Retrieve(object primaryKey)
    {
        return DbSet.Find(primaryKey);
    }

Метод DbSet.Find :

Находит сущность с заданными значениями первичного ключа. Если сущность с данные значения первичного ключа существуют в контексте, то это вернулся сразу, не сделав запрос в магазин. Иначе, в магазин поступает запрос на объект с указанным основным ключевые значения, и этот объект, если он найден, присоединяется к контексту и вернулся. Если в контексте или хранилище не найдено ни одного объекта, тогда null возвращается.

По сути, он возвращает прикрепленный объект данного primaryKey, поэтому вам просто нужно применить изменения к возвращенному объекту, чтобы сохранить правильный экземпляр.

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