在使用Entity Framework作为ORM来存取数据的过程中,最常规的操作就是对数据对象的更新。本文将会包含如何Attach Entity到一个数据Context中,以及如何使用EntityState字段状态来保存数据变化。

文本参考了如下两篇文章:

https://msdn.microsoft.com/en-us/data/jj592676

https://stackoverflow.com/questions/30987806/dbset-attachentity-vs-dbcontext-entryentity-state-entitystate-modified

当要做数据更新时,我们通常会使用如下代码:

context.Entity(entity).State = EntityState.Modified;

context.SaveChanges();

context.Entity(entity).State = EntityState.Modified; 这句代码不仅会将entity attaching到当前的context中,而且还会告诉context这个entity已经发生了变化。当执行savechanges()时,EF会使用update语句将entity中所有的字段全部更新。

上面的这种方式虽然方便,但是在某些时候也会带来效率上的浪费。如果我们只是想更新其中的某个或者某些字段而不是全部字段的时候,使用这种方式就不必要了。那么该如何操作,才能让Entity Framework精确的保存我们想要变更的字段呢?我们可以使用Dbset.Attach(entity)。

Dbset.Attach(entity)只是会将entity attach到context中,而不会告诉context这个实体已经发生修改。所以如果你只是attach entity而之后不做任何处理的话,savechanges()操作将不会有任何变化。所以它和context.Entity(entity).State = EntityState.Unchanged是相同的。要保存个别字段,需要将entity更改之后再调用savechanges方法。请参见下面的代码样例。

代码样例:

下面使我们定义的实体以及它们的Context:

 public class Department
{
public int Id {get;set;}
public string Name {get;set;}
public DateTime CreateTime {get;set;} public int DepLeadID { get; set; } [ForeignKey("DepLeadID")]
public DepLead Lead {get;set;}
}
 public class DepLead
{
public int Id {get;set;}
public string name {get;set;}
public int age {get;set;}
}
 public class Context : DbContext
{
public Context() : base("XXX")
{
}
public DbSet<Department> Departments { get; set; }
public DbSet<DepLead> DepLeads { get; set; } protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Conventions.Remove<PluralizingTableNameConvention>();
} }

如果我们使用如下方式更新数据:

context.Entity(someDepartment).State = EntityState.Modified;
context.SaveChanges();

那么相当于执行如下SQL

UPDATE department
SET Name = 'xxx',CreateTime = 'XXX', DepLeadID = 'XXX'
where Id = '1000'; -- the id of someDepartment

上面的这种方式,不会自动的检查字段是否变化,而是把所有的字段全部按照someDepartment中的值进行更新,即使没有任何变化。

而另外一种Attach的方式在更新上就会更加灵活。

context.Departments.Attach(someDepartment);
someDepartment.Name = 'XXX';
context.SaveChanges();

以上片段会执行下面语句:

UPDATE department
set Name = 'XXX'
where Id = '1000'

假如你想指定实体someDepartment中哪些字段是已经更改的状态,可以按如下代码:

context.Departments.Attach(someDepartment);
context.Entry(someDepartment).Property("Name").IsModified = true;
context.SaveChanges();

这样它也会像上面的SQL那样将Name字段更新。

Entity Framework使用EntityState和Attach来保存数据变化以及更新实体的个别字段的更多相关文章

  1. entity framework 6 我写了一个公用数据类

    public class BaseDAL { string strConn = ""; public BaseDAL(string connString) { strConn = ...

  2. 《Entity Framework 6 Recipes》中文翻译系列 目录篇 -持续更新

    为了方便大家的阅读和学习,也是响应网友的建议,在这里为这个系列做一个目录.在目录开始这前,我先来回答之前遇到的几个问题. 1.为什么要学习EF? 这个问题很简单,项目需要.这不像学校,没人强迫你学习! ...

  3. 《Entity Framework 6 Recipes》翻译系列 (1) -----第一章 开始使用实体框架之历史和框架简述

    微软的Entity Framework 受到越来越多人的关注和使用,Entity Framework7.0版本也即将发行.虽然已经开源,可遗憾的是,国内没有关于它的书籍,更不用说好书了,可能是因为EF ...

  4. 《Entity Framework 6 Recipes》翻译系列(2) -----第一章 开始使用实体框架之使用介绍

    Visual Studio 我们在Windows平台上开发应用程序使用的工具主要是Visual Studio.这个集成开发环境已经演化了很多年,从一个简单的C++编辑器和编译器到一个高度集成.支持软件 ...

  5. Entity FrameWork对有外键关联的数据表的添加操作

    前天做了一个MVC Entity FrameWork项目,遇到有外键关联的数据编辑问题.当你编辑的时候,按照正常的逻辑,把每个字段的数据都对号入座了,然后点击保存按钮,本以为会顺理成章的编辑数据,但是 ...

  6. Working with Data » Getting started with ASP.NET Core and Entity Framework Core using Visual Studio » 读取关系数据

    Reading related data¶ 9 of 9 people found this helpful The Contoso University sample web application ...

  7. Entity Framework 使用注意:Where查询条件中用到的关联实体不需要Include

    来自博客园开发团队开发前线最新消息: 在Entity Framework中,如果实体A关联了实体B,你想在加载实体A的同时加载实体B.通常做法是在LINQ查询中使用Include().但是,如果你在查 ...

  8. Entity Framework:三种开发模式实现数据访问

    原文地址 http://blog.csdn.net/syaguang2006/article/details/19606715 前言 Entity Framework支持Database First. ...

  9. Entity Framework(三):使用特性(数据注解)创建表结构

    一.理解Code First及其约定和配置 传统设计应用的方式都是由下而上的,即我们习惯优先考虑数据库,然后使用这个以数据为中心的方法在数据之上构建应用程序.这种方法非常适合于数据密集的应用或者数据库 ...

随机推荐

  1. MySQL数据库二

    筛选条件 比较运算符: 等于: =  (注意!不是==)            大于等于: >=          IS NULL 不等于: !=  或  <>        小于: ...

  2. 简单分析ucenter 会员同步登录通信原理

    1.用户登录discuz,通过logging.php文件中的函数uc_user_login对post过来的数据进行验证,也就是对username和password进行验证. 2.如果验证成功,将调用位 ...

  3. go 反射包

    一.什么是反射? 反射是用程序检查其所拥有的结构,尤其是类型的一种能力: 二.Printf Printf 的函数声明为: func Printf(format string, args ... int ...

  4. redis 练习 a的数据库数据迁移到b数据库

    思路 1.从a redis中获取所有的key 2.判断key的类型 3.根据key的类型,判断使用的是set/hset类型 4.set到b redis中(写入到b redis中)

  5. 127.0.0.1和localhost区别

  6. Docker-CentOS系统安装Docker

    上一节,我们介绍了安装虚拟机及操作系统,本文再详细描述安装docker的命令. 前提条件 虚拟机系统:CentOS,并且虚拟机能连通外网. 另外,虚拟机最好配置上阿里的镜像源,点此链接,进入cento ...

  7. GitHub 热点速览 Vol.16:化身蒙娜丽莎和乔布斯对话

    摘要:妙趣横生,上周的 GitHub 热点的关键词.无论是让你化身为爱因斯坦开启会议脑暴模式 avatarify,还是和上周人人都是抠图师项目的同门项目 3D 照片修复:3d-photo-inpain ...

  8. 嘿嘿,我就知道面试官接下来要问我 ConcurrentHashMap 底层原理了,看我怎么秀他

    前言 上篇文章介绍了 HashMap 源码后,在博客平台广受好评,让本来己经不打算更新这个系列的我,仿佛被打了一顿鸡血.真的,被读者认可的感觉,就是这么奇妙. 然后,有读者希望我能出一版 Concur ...

  9. redis和memcache列出所有key

    //redis $redis = new Redis(); $redis->connect("host", "port"); $redis->sel ...

  10. PHP出现SSL certificate:unable to get local issuer certificate的解决办法

    当本地curl需要访问https时,如果没有配置证书,会出现SSL certificate: unable to get local issuer certificate错误信息. 解决办法: 1.下 ...