实体框架6:除ID以外的克隆对象

时间:2014-09-08 09:12:45

标签: c# entity-framework mvvm entity-framework-6

在我的MVVM程序中,我有一个Model类(比如MyModel),我从中有一个从数据库中读取的实例(使用Entity Framework)。在检索对象时,我将所有数据呈现给用户。稍后用户将修改一些字段 我想要的是创建相同的对象,除了它ID(因为ID主键自动递增)。 /> 那我怎么能接近这个呢?我不想一个接一个地复制所有字段,这不是一个强大的方法。因为将来可能会修改模型,所以我必须在克隆方法中考虑到这一点。

那么有没有优雅的方法来复制对象,当保存在数据库中时,它的ID会再次自动递增? (将ID设置为null会给我一个编译器错误,因为它的类型为int)。

5 个答案:

答案 0 :(得分:52)

我注意到没有必要复制。显然,在向数据库添加模型实例时(即使ID设置为数据库中已存在的ID),实体框架会在数据库中插入一个新行并自动递增它的主键。所以这个功能已经内置到EF中。抱歉,我不知道这件事 为了清楚起见,这是一个例子:

using(var database = new MyDbContext()) {
    MyModel myModel = database.FirstOrDefault(m => m.SomeProperty == someValue);
    myModel.SomeOtherProperty = someOtherValue; //user changed a value
    database.MyModels.Add(myModel); //even though the ID of myModel exists in the database, it gets added as a new row and the ID gets auto-incremented 
    database.SaveChanges();
}

答案 1 :(得分:35)

Lori Peterson建议使用.AsNoTracking()在EF6中执行克隆。我正在使用这种方法,可以确认它有效。您甚至可以包含子对象。

var entity = context.Entities
                    .AsNoTracking()
                    .Include(x => x.ChildEntities)
                    .FirstOrDefault(x => x.EntityId == entityId);

entity.SomeProperty = DateTime.Now;

context.Entities.Add(entity);
context.SaveChanges();
  

当您从数据集中检索一个或多个实体时,您可以告诉实体框架不要跟踪您对该对象所做的任何更改,然后将该实体作为新实体添加到数据集中。使用.AsNoTracking时,上下文对现有实体一无所知。

答案 2 :(得分:4)

使用ObjectContext时,QuantumHive提供的答案不起作用。

在这种情况下返回的错误是:

An object with the same key already exists in the ObjectStateManager. The ObjectStateManager cannot track multiple objects with the same key.
System.InvalidOperationException: An object with the same key already exists in the ObjectStateManager. The ObjectStateManager cannot track multiple objects with the same key.
   at System.Data.Objects.ObjectStateManager.AddEntry(IEntityWrapper wrappedObject, EntityKey passedKey, EntitySet entitySet, String argumentName, Boolean isAdded)
   at System.Data.Objects.ObjectContext.AddSingleObject(EntitySet entitySet, IEntityWrapper wrappedEntity, String argumentName)
   at System.Data.Objects.DataClasses.RelatedEnd.AddEntityToObjectStateManager(IEntityWrapper wrappedEntity, Boolean doAttach)
   at System.Data.Objects.DataClasses.RelatedEnd.AddGraphToObjectStateManager(IEntityWrapper wrappedEntity, Boolean relationshipAlreadyExists, Boolean addRelationshipAsUnchanged, Boolean doAttach)
   at System.Data.Objects.DataClasses.RelatedEnd.Add(IEntityWrapper wrappedTarget, Boolean applyConstraints, Boolean addRelationshipAsUnchanged, Boolean relationshipAlreadyExists, Boolean allowModifyingOtherEndOfRelationship, Boolean forceForeignKeyChanges)
   at System.Data.Objects.DataClasses.RelatedEnd.Add(IEntityWrapper wrappedEntity, Boolean applyConstraints)
   at System.Data.Objects.DataClasses.EntityReference`1.set_ReferenceValue(IEntityWrapper value)
   at System.Data.Objects.DataClasses.EntityReference`1.set_Value(TEntity value)

正确克隆实体框架对象(至少在EF6.0中)是:

/// <summary>
/// Clone a replica of this item in the database
/// </summary>
/// <returns>The cloned item</returns>
public Item CloneDeep()
{
    using (var context = new EntityObjectContext())
    {
        var item = context.Items
            .Where(i => i.ItemID == this.ItemID)
            .Single();
        context.Detach(item);
        item.EntityKey = null;
        item.ItemID = 0;
        return item;
    }
}

答案 3 :(得分:4)

我发现这样看是否有更好的克隆对象的方法比我目前使用的更好,并注意到如果你想要做多个克隆,接受的答案可能存在问题...至少如果你想避免多次创建你的上下文......

我不知道这是否是克隆的最佳方法,这就是我寻找另一种方式的原因。但是,它有效。如果需要多次克隆实体,可以使用JSON序列化来克隆......这样的事情(使用Newtonsoft JSON)。

using( var context = new Context() ) {
    Link link    = context.Links.Where(x => x.Id == someId);
    bool isFirst = true;
    foreach( var id in userIds ) {
        if( isFirst ) {
            link.UserId = id;
            isFirst     = false;
        }
        else {
            string cloneString = JsonConvert.SerializeObject(link);
            Link clone = JsonConvert.DeserializeObject<Link>(cloneString);
            clone.UserId = id;
            context.Links.Add(clone);
        }
    }
    context.SaveChanges();
}

答案 4 :(得分:0)

我使用postgres数据库:

CREATE TABLE public."Table" ( 
    "Id" integer NOT NULL DEFAULT nextval('"Table_Id_seq"'::regclass),
    ...

在我的情况下,没有一种方法无法使用。我第二次使用:

Table table = _context.Table.AsNoTracking().Select(s => new Table {
 // some properties, exept id
    }).FirstOrDefault();
_context.Table.Add(table);
await _context.SaveChangesAsync();