上一篇我们介绍了Entity Framework Core系列之DbContext(添加),这一篇我们介绍下修改数据

修改实体的方法取决于context是否正在跟踪需要修改的实体。

下面的示例中实体由context获得,所以context会开始追踪这个实体。当我们更改这个实体的属性值时,context会将实体的EntityState更改为已修改,ChangeTracker将记录旧属性值和新属性值。当调用SaveChanges时,会生成update语句并执行。

 var author = context.Authors.First(a => a.AuthorId == );
author.FirstName = "Bill";
context.SaveChanges();

由于ChangeTracker记录哪些属性已被修改,Context将发出一条SQL语句,该语句只更新已修改的属性:

exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Authors] SET [FirstName] = @p0
WHERE [AuthorId] = @p1;
SELECT @@ROWCOUNT;
',N'@p1 int,@p0 nvarchar(4000)',@p1=1,@p0=N'Bill'

断开连接的场景

在一个断开连接的场景如ASP.NET应用程序中,在controller或者服务方法中可能会发生更改现有实体属性值。在这种情况下,需要告知Context实体处于修改状态, 有几种方法: 显式地为实体设置EntityState; 用DbContext.Update 方法(EF Core新内容); 使用DbContext.Attach方法,然后“遍历对象图”,以显式地设置图中各个属性的状态.

设置EntityState

通过EntityEntry设置实体的EntityState

 public void Save(Author author)
{
using (var context = new EFCoreContext())
{
context.Entry(author).State = EntityState.Modified;
context.SaveChanges();
}
}

这种方法只会导致Author实体被设置成修改后的状态。不会设置任何相关对象。由于ChangeTracker不知道修改了哪些属性,因此Context生成一条SQL语句来更新所有属性值(除了主键值之外)。

DbContext Update

DbContext类提供用于处理单个或多个实体的Update和UpdateRange方法。

 public void Save(Author author)
{
using (var context = new EFCoreContext())
{
context.Update(author);
context.SaveChanges();
}
}

与设置EntityState方法一样,这个方法也会将Context追踪的实体设置成修改状态。同样,Context没有任何方法来识别哪些属性值已经更改,所以生成SQL来更新所有属性。与显示设置设置EntityState不同的是,Context也会修改相关实体(如本例中的Books)的状态为已修改,从而会为每个实体生成update语句。如果相关实体没有对应的键值,就会标记为add,生成一条Insert语句。

DbContext Attach

当在实体上使用Attach方法时,它的状态将被设置为unchanged,这将导致根本不会生成任何sql语句,所有定义了键值的其他可访问实体也将被设置为unchanged。那些没有键值的将被标记为Added。但是,现在该实体正在被Context跟踪,我们可以通知上下文哪些属性被修改,生成正确的UPDATE SQL:

 using (var context = new EFCoreContext())
{
var author = new Author
{
AuthorId = ,
FirstName = "yixuan",
LastName = "han"
};
author.Books.Add(new Book { BookId = , Title = "Othello" }); context.Attach(author);
context.Entry(author).Property("FirstName").IsModified = true;
context.SaveChanges();
}

上面的代码会将Author实体设置成被修改,生成的sql语句也之后更新FirstName。

exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Authors] SET [FirstName] = @p0
WHERE [AuthorId] = @p1;
SELECT @@ROWCOUNT;
',N'@p1 int,@p0 nvarchar(4000)',@p1=1,@p0=N'William'

TrackGraph

TrackGraph API提供了对对象图中各个实体的访问,并允许您对每个实体分别执行定制代码。这在处理这种由不同对象的相关实体的复杂对象图的场景中非常有用。下面的示例复制了一个场景,其中对象图是在Context之外构造的。然后用TrackGraph方法“走图”:

   var author = new Author
{
AuthorId = ,
FirstName = "yixuan",
LastName = "han"
};
author.Books.Add(new Book { AuthorId = , BookId = , Title = "Hamlet", Isbn = "" });
author.Books.Add(new Book { AuthorId = , BookId = , Title = "Othello", Isbn = "" });
author.Books.Add(new Book { AuthorId = , BookId = , Title = "MacBeth", Isbn = "" });
using (var context = new EFCoreContext())
{
context.ChangeTracker.TrackGraph(author, e => {
if ((e.Entry.Entity as Author) != null)
{
e.Entry.State = EntityState.Unchanged;
}
else
{
e.Entry.State = EntityState.Modified;
}
});
context.SaveChanges();
}

在这个场景中,假定Author实体没有被修改,但是Book可能被编辑过。TrackGraph方法将根实体作为参数,并使用lambda指定要执行的操作。在这种情况下,根实体(Author)将其EntityState设置为不变。Context开始跟踪实体,设置EntityState,只有这样才能发现和它相关的实体。Books的EntityState设置为Modified,与前面的示例一样,这将导致生成SQL更新实体上的所有属性:

exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Books] SET [AuthorId] = @p0, [Isbn] = @p1, [Title] = @p2
WHERE [BookId] = @p3;
SELECT @@ROWCOUNT;
UPDATE [Books] SET [AuthorId] = @p4, [Isbn] = @p5, [Title] = @p6
WHERE [BookId] = @p7;
SELECT @@ROWCOUNT;
UPDATE [Books] SET [AuthorId] = @p8, [Isbn] = @p9, [Title] = @p10
WHERE [BookId] = @p11;
SELECT @@ROWCOUNT;
',N'@p3 int,@p0 int,@p1 nvarchar(4000),@p2 nvarchar(150),@p7 int,@p4 int,@p5 nvarchar(4000),
@p6 nvarchar(150),@p11 int,@p8 int,@p9 nvarchar(4000),@p10 nvarchar(150)',
@p3=1,@p0=1,@p1=N'',@p2=N'Hamlet',
@p7=2,@p4=1,@p5=N'',@p6=N'Othello',
@p113,@p8=1,@p9=N'',@p10=N'MacBeth'

因为SQL更新了所有属性,所以它们都需要显示并分配一个有效值,否则它们将被更新为默认值。在下一个示例中,对象图再次在Context之外构造,但只修改了books的Isbn属性。因此,其他属性(除了实体键)被省略:

 var author = new Author
{
AuthorId = ,
FirstName = "William",
LastName = "Shakespeare"
};
author.Books.Add(new Book { BookId = , Isbn = "" });
author.Books.Add(new Book { BookId = , Isbn = "" });
author.Books.Add(new Book { BookId = , Isbn = "" }); using (var context = new EFCoreContext())
{
context.ChangeTracker.TrackGraph(author, e => {
e.Entry.State = EntityState.Unchanged; //starts tracking
if ((e.Entry.Entity as Book) != null)
{
context.Entry(e.Entry.Entity as Book).Property("Isbn").IsModified = true;
}
});
}

这一次,lambda的方法主体确保所有实体都被设置成Unchanged,然后指示Isbn属性被修改。这导致生成的SQL只更新Isbn属性值:

exec sp_executesql N'SET NOCOUNT ON;
UPDATE [Books] SET [Isbn] = @p0
WHERE [BookId] = @p1;
SELECT @@ROWCOUNT;
UPDATE [Books] SET [Isbn] = @p2
WHERE [BookId] = @p3;
SELECT @@ROWCOUNT;
UPDATE [Books] SET [Isbn] = @p4
WHERE [BookId] = @p5;
SELECT @@ROWCOUNT;
',N'@p1 int,@p0 nvarchar(4000),@p3 int,@p2 nvarchar(4000),@p5 int,@p4 nvarchar(4000)',
@p1=1,@p0=N'',
@p3=2,@p2=N'',
@p5=3,@p4=N''

Entity Framework Core系列之DbContext(修改)的更多相关文章

  1. Entity Framework Core系列之DbContext(删除)

    上一篇我们介绍了Entity Framework Core系列之DbContext(修改),这一篇我们介绍下删除数据 修改实体的方法取决于context是否正在跟踪需要删除的实体. 下面的示例中con ...

  2. Entity Framework Core系列之DbContext

    前言: EF Core DbContext表示与数据库的会话,并提供与数据库通信的API,具有以下功能: 数据库连接 数据操作,如查询和持久化 更改追踪 模型构建 数据映射 对象缓存 事务管理 数据库 ...

  3. Entity Framework Core系列之DbContext(添加)

    上一篇我们介绍了Entity Framework Core系列之DbContext,对DbContext有了概念上的了解,这篇将介绍DbContext添加数据 通过DbContext添加实体的主要方法 ...

  4. Entity Framework Core系列之什么是Entity Framework Core

    前言 Entity Framework Core (EF Core)是微软推荐的基于.NET Core framework的应用程序数据访问技术.它是轻量级,可扩展并且支持跨平台开发.EF Core是 ...

  5. Entity Framework Core(3)-配置DbContext

    设计时 DbContext 配置 EF Core 设计时工具如迁移需要能够发现和创建的工作实例DbContext以收集有关应用程序的实体类型以及它们如何映射到数据库架构的详细信息的类型. 此过程可以为 ...

  6. Entity Framework Core系列之实战(ASP.NET Core MVC应用程序)

    本示例演示在ASP.NET 应用程序中使用EF CORE创建数据库并对其做基本的增删改查操作.当然我们默认你的机器上已经安装了.NET CORE SDK以及合适的IDE.本例使用的是Visual St ...

  7. ASP.NET CORE系列【六】Entity Framework Core 之数据迁移

    原文:ASP.NET CORE系列[六]Entity Framework Core 之数据迁移 前言 最近打算用.NET Core写一份简单的后台系统,来练练手 然后又用到了Entity Framew ...

  8. ASP.NET CORE系列【六】Entity Framework Core 之数据库迁移

    前言 最近打算用.NET Core写一份简单的后台系统,来练练手 然后又用到了Entity Framework Core 发现园子里有些文章讲得不是那么细节,对于新手小白来说,可能会有点懵. 特意整理 ...

  9. 002从零开始入门Entity Framework Core——DbContext生存期、配置和初始化

    阅读须知:本文为入门介绍.指引文章,所示代码皆为最简易(或仅为实现功能)的演示示例版本,不一定切实符合个人(企业)实际开发需求. 一.DbContext生存期 DbContext 的生存期从创建实例时 ...

随机推荐

  1. .Net Framework 4.5.1 ASP.NET MVC 5 下新建视图报“错误 运行所选代码生成器时出错 无法检索元数据 没有为该对象定义无参数构造函数”

    当在控制器中新建视图的时候,选择的视图界面如下: 执行添加后报如下错误: 错误的内容为: 错误运行所选代码生成器时出错 无法检索"XXX"的元数据没有为该对象定义无参数构造函数 U ...

  2. 从零开始学安全(三十七)●VM汇编环境搭建

    需要下载 vm 虚拟机 破解版连接 链接:https://pan.baidu.com/s/1r9MyPkNBmiYhQ8bdUxPmvQ 提取码:2o98 镜像文件和开发环境 链接:https://p ...

  3. 从URL到看到网页的过程

    从我们输入URL并按下回车键到看到网页结果之间发生了什么?换句话说,一张网页,要经历怎样的过程,才能抵达用户面前?下面来从一些细节上面尝试一下探寻里面的秘密. 前言:键盘与硬件中断 说到输入URL,当 ...

  4. 前端项目git操作命名规范和协作开发流程

    前言 一个项目的分支,一般包括主干 master 和 开发分支 dev,以及若干临时分支 分支命名规范 分支: 命名: 说明: 主分支 master 主分支,所有提供给用户使用的正式版本,都在这个主分 ...

  5. html/css的学习之路(1)

    HTML5简介:HTML5是什么?要回答这个问题,我们需要先了解一下HTML是什么.HTML的英文全称为Hyper Text Markup Language,即超文本标记语言.HTML5是HTML的一 ...

  6. JS数组Sort方法的使用

    想用sort方法对数组排下序,代码如下: var nums = "12 645 6 85 81 0 9 365 4 752".split(" ").map(fu ...

  7. AI时代大点兵-国内外知名AI公司2018年最新盘点

    AI时代大点兵-国内外知名AI公司2018年最新盘点 导言 据腾讯研究院统计,截至2017年6月,全球人工智能初创企业共计2617家.美国占据1078家居首,中国以592家企业排名第二,其后分别是英国 ...

  8. ORACLE如何检查找出损坏索引(Corrupt Indexes)

      在Oracle数据库中如何找出损坏索引呢? 下面我们人为构造一个案例,将索引块损坏.如下案例所示: SQL> create tablespace test_data   2  datafil ...

  9. selenium-确定找到的element唯一(三)

    在python + selenium 中经常会遇到找到的元素不唯一,导致定位到的元素不是预期的或者定位不到元素 解决方法:只要在页面进行确认找到的元素唯一后,再进行操作 页面确认方法: 1.通过htm ...

  10. 不能收缩 ID 为 %s 的数据库中 ID 为 %s 的文件,因为它正由其他进程收缩或为空。

    SQLServer数据库通常都不建议进行SHRINKFILE操作,因为SHRINKFILE不当会造成一定的性能问题. 但是当进行了某些操作(例如某个超大的日志类型表转成分区表切换了数据文件),数据库某 ...