本文介绍了从更新时调用SaveChanges()被调用排除属性的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

似乎有两种方法来更新使用连接的方法断开连接的实体框架的实体。

方法一,您只需设置断开实体的状态修改:

  myDbContext.Dogs.Attach(狗);
myDbContext.Entry(狗).STATE = EntityState.Modified;
myDbContext.SaveChanges();

这将保存狗的对象的所有领域。但是,假设你是从一个MVC的Web页面,您只允许Dog.Name的编辑这样做,并包含在页面上唯一的狗属性名称。那么可以做方法二:

  myDbContext.Dogs.Attach(狗);
myDbContext.Entry(狗).Property(O = GT; o.Name).CurrentValue = dog.Name;
myDbContext.Entry(狗).Property(O = GT; o.Name).IsModified = TRUE;
myDbContext.SaveChanges();

方法二可以得到相当冗长的时候有很多的属性来更新。这促使我尝试方法三,设置将IsModified =上我不想改变的属性假的。这不起作用,抛出运行时错误设置将IsModified为false,修改后的财产不被支持

  myDbContext.Dogs.Attach(狗);
myDbContext.Entry(狗).STATE = EntityState.Modified;
myDbContext.Entry(狗).Property(O = GT; o.Owner).IsModified = FALSE;
myDbContext.SaveChanges();

我更preFER使用方法一无处不在,但也有在我的asp.net的MVC视图不包含狗类的每个标量属性许多实例。

我的问题是:


  1. 有我可以在POCO类会告诉我再也不想财产最多更新实体框架使用的属性?例如,[NeverUpdate]。我知道的[NotMapped]属性的,但是这不是我所需要的。

  2. 如果做不到这一点,有没有什么办法,我可以使用方法一以上(myDbContext.Entry(狗).STATE = EntityState.Modified;
    ),并排除我不想要更新的字段?

P.S。我知道的另一种方式,不使用附加,只是从数据库中获取一个新的对象,更新所需的性能,并保存。这就是我在做什么,但我很好奇,如果有使用的方式连接,从而避免了额外的旅行到数据库,但这样做的方式,不那么冗长如上法两种。通过获取一个新的对象,我的意思是:

 狗dbDog = myDbContext.Dogs.FirstOrDefault(D => d.ID = dog.ID);
dbDog.Name = dog.Name;
myDbContext.SaveChanges();


解决方案

下面的工作

  myDbContext.Dogs.Attach(狗);
myDbContext.Entry(狗).STATE = EntityState.Modified;VAR的ObjectContext =((IObjectContextAdapter)myDbContext).ObjectContext;
的foreach(在objectContext.ObjectStateManager.GetObjectStateEntries(EntityState.Modified)。凡(实体=&GT VAR进入; entity.Entity.GetType()== typeof运算(狗)))
{
    //你需要给外键属性名
    //而不是导航属性名
    entry.RejectPropertyChanges(OWNERID);}myDbContext.SaveChanges();


如果你想要做它在一个单一的线,使用下面的扩展方法:

 公共静态无效DontUpdateProperty< TEntity>(这方面的DbContext,弦乐propertyName的)
{
    变种的ObjectContext =((IObjectContextAdapter)上下文).ObjectContext;
    的foreach(在objectContext.ObjectStateManager.GetObjectStateEntries(EntityState.Modified)。凡(实体=&GT VAR进入; entity.Entity.GetType()== typeof运算(TEntity)已))
    {
        entry.RejectPropertyChanges(propertyName的);
    }
}

和使用它像这样

  //在修改了一些波苏斯
myDbContext.DontUpdateProperty<狗狗和GT(OWNERID);
myDbContext.SaveChanges();

正如你所看到的,你可以修改此解决方案,以满足您的需求,例如,使用的String []属性而不是字符串propertyName的作为参数。


建议的方法

有一个更好的解决办法是如你所说([NeverUpdate])使用一个属性。为了使它工作,你需要使用SavingChanges事件(检查我的<一个href=\"https://heli$c$cr.word$p$pss.com/2013/07/20/execute-$c$c-before-save-and-after-fetch-in-entity-framework/\"相对=nofollow称号=博客>博客):

 无效ObjectContext_SavingChanges(对象发件人,System.Data.Objects.SavingChangesEventArgs E)
{
    ObjectContext的背景=发件人为ObjectContext的;
    如果(上下文!= NULL)
    {
        的foreach(在context.ObjectStateManager.GetObjectStateEntries ObjectStateEntry条目(EntityState.Modified))
        {
            VAR类型= ty​​peof运算(entry.Entity);
            变种属性= type.GetProperties();
            的foreach(在性能VAR属性)
            {
                VAR属性= property.GetCustomAttributes(typeof运算(NeverUpdateAttribute),FALSE);
                如果(attributes.Length大于0)
                    entry.RejectPropertyChanges(property.Name);
            }
        }
    }
}
//检查如何创建自定义属性Microsoft文档:
// http://msdn.microsoft.com/en-us/library/sw480ze8(v=vs.80).aspx
公共类NeverUpdateAttribute:SystemAttribute
{}//在你的POCO
公共类犬
{
    [NeverUpdate]
    公众诠释OWNERID {搞定;组; }
}


报警2:我刚才看了 MSDN文档和它说:


Warning 3: I have tried it now. This solution works. Property that is rejected with RejectPropertyChanges() method are not updated in the persistence unit (database).

HOWEVER, if the entity that is updated is attached by calling Attach(), the current context remains dirty after SaveChanges(). Assume that the following row exists in the database:

Dogs
ID: 1
Name: Max
OwnerID: 1

Consider the following code:

var myDog = new Dogs();
myDog.ID = 1;
myDog.Name = Achilles;
myDog.OwnerID = 2;

myDbContext.Dogs.Attach(myDog);
myDbContext.Entry(myDog).State = EntityState.Modified;
myDbContext.SaveChanges();

The current state of database after SaveChanges():

Dogs:
ID: 1
Name: Achilles
OwnerID: 1

The current state of myDbContext after SaveChanges():

var ownerId = myDog.OwnerID;  // it is 2
var status = myDbContext.Entry(myDog).State; // it is Unchanged

So what you should do? Detach it after SaveChanges():

Dogs myDog = new Dogs();
//Set properties
...
myDbContext.Dogs.Attach(myDog);
myDbContext.Entry(myDog).State = EntityState.Modified;
myDbContext.SaveChanges();
myDbContext.Entry(myDog).State = EntityState.Detached;

这篇关于从更新时调用SaveChanges()被调用排除属性的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!

08-18 11:26