Entity Framework Code First实现乐观并发
Entity Framework Code First实现乐观并发
不定时更新翻译系列,此系列更新毫无时间规律,文笔菜翻译菜求各位看官老爷们轻喷,如觉得我翻译有问题请挪步原博客地址
本博文翻译自:
https://www.codeproject.com/Articles/817432/Optimistic-Concurrency-in-Entity-Framework-Code-Fi
介绍
本文描述了使用Entity Framework Code First处理乐观并发性的不同配置
并发性
在计算机科学中,并发性是系统的一个属性,在这个系统中,多个计算同时执行,并且有可能相互影响。
在web应用程序中,这是一个多用户环境,在保存数据库中的数据时,可能存在并发性。并发性大致分为两种类型:1)悲观并发2)乐观并发
1) 悲观并发
数据库中的悲观并发包括锁行,以防止其他用户以影响当前用户的方式修改数据。
在这种方法中,用户执行一个操作,其中一个锁被应用,其他用户不能在该记录上执行相同的操作,直到该锁被释放。
2) 乐观并发
相比之下,在乐观并发中,当用户阅读时,行不会被锁定。当用户试图更新这一行时,系统必须确定该记录是否被另一个用户修改过,因为它被读取了。
开始编写代码
让我们创建一个控制台应用程序来探索处理乐观并发的不同情况。
步骤
- 使用Visual Studio,创建控制台应用程序(文件 ->新建->项目->控制台程序(使用 Visual C# 模板)) 并将其命名为 ConcurrencyCheck.
- 向项目添加一个新的文件夹 Models. 然后在这个文件夹中添加两个类文件: EducationContext.cs 和 Student.cs .
- 在这个控制台应用程序中安装EntityFramework Nuget包。在包管理器控制台中运行"_Install-Package EntityFramework"_命令来执行此操作。或者,您也可以"Nuget包管理器"理窗口安装相同的程序。
下表显示了用于乐观并发的不同配置。
配置乐观并发
| Convention | None |
|---|---|
| Data Annotation | [Timestamp] |
| Fluent API | .IsRowVersion() |
1) Convention
Entity Framework Code First 没有任何处理乐观并发的约定。您可以使用Data Annotation或Fluent API来处理乐观并发。
2) Data Annotation
Code First使用**[Timestamp] **处理乐观并发性的属性。
a) 修改 EducationContext.cs 文件如下:
using System.Data.Entity;
namespace ConcurrencyCheck.Models
{
class EducationContext : DbContext
{
public EducationContext()
: base("EducationContext")
{
}
public DbSet<Student> Students { get; set; }
}
}
在base("EducationContext")中通过Code First 指令在 App.config 文件中使用名为"EducationContext"的连接字符串
b) 修改 Student.cs 文件如下:
using System.ComponentModel.DataAnnotations;
namespace ConcurrencyCheck.Models
{
public class Student
{
public int StudentId { get; set; }
public string RollNumber { get; set; }
public string FirstName { get; set; }
public string LastName { get; set; }
[Timestamp]
public byte[] RowVersion { get; set; }
}
}
请注意,在 Student 类中有一个属性 RowVersion ,它是 byte[] 类型,并被分配使用 [Timestamp] 属性来处理乐观的并发性。
c) 将 App.config 文件中的连接字符串更改为指向一个有效的数据库:
<connectionStrings>
<add name="EducationContext" providerName="System.Data.SqlClient" connectionString="Server=DUKHABANDHU-PC; Database=ConcurrencyCheck;Integrated Security=SSPI" />
</connectionStrings>
在这里,我们将数据库名称作为ConcurrencyCheck,它将在应用程序运行时通过 Code First 创建。
d) 修改 Program.cs 文件在每次应用程序运行时都要删除和创建数据库:
static void Main(string[] args)
{
Database.SetInitializer(new DropCreateDatabaseAlways<EducationContext>());
using (var context = new EducationContext())
{
context.Students.Add(new Student
{
FirstName = "Dukhabandhu",
LastName = "Sahoo",
RollNumber = "1"
});
context.SaveChanges();
}
Console.WriteLine("Database Created!!!");
Console.ReadKey();
}
如果运行该应用程序,代码首先将创建数据库 ConcurrenCheck 它拥有两个表 MigrationHistory 和 Students 。
如果您看到 Students 表中的 RowVersion 列(在SQL Sever中),它的数据类型是 timestamp 。

RowVersion 和 TimeStamp 是不同数据库提供程序用于相同目的的两个术语。当创建或更新 Students 表中的记录时,数据库将自动更新 RowVersion 值到新值。即使您为 rowversion 列发送值,数据库(SQL Server)也不使用该值来进行插入或更新操作。
当添加到 Students 表的新记录时生成的SQL:
exec sp_executesql N'INSERT [dbo].[Students]([RollNumber], [FirstName], [LastName])
VALUES (@0, @1, @2)
SELECT [StudentId], [RowVersion]
FROM [dbo].[Students]
WHERE @@ROWCOUNT > 0 AND [StudentId] = scope_identity()',N'@0 nvarchar(max) ,@1 nvarchar(max) ,@2 nvarchar(max) ',@0=N'1',@1=N'Dukhabandhu',@2=N'Sahoo'
您可以看到查询不仅插入了一个新记录,而且还返回了 RowVersion 的值。
当进行更新和删除操作时,会发生实际的并发检查。在更新和删除 Students 表的记录时,请参阅下面如何发生并发检查。
UPDATE SQL
exec sp_executesql N'UPDATE [dbo].[Students]
SET [RollNumber] = @0 WHERE (([StudentId] = @1) AND ([RowVersion] = @2))
SELECT [RowVersion] FROM [dbo].[Students]
WHERE @@ROWCOUNT > 0 AND [StudentId] = @1',N'@0 nvarchar(max) ,@1 int,@2 binary(8)',@0=N'2',@1=1,@2=0x00000000000007D1
我们看到 WHERE 条件,在更新记录时,它比较了 studentid (主键)和 RowVersion 。
DELETE SQL
exec sp_executesql N'DELETE [dbo].[Students]
WHERE (([StudentId] = @0) AND ([RowVersion] = @1))',N'@0 int,@1 binary(8)',@0=1,@1=0x00000000000007D1
在删除记录代码之前,先创建一个查询来比较标识符(主键 StudentId )和行版本( RowVersion 字段)用于乐观并发。
3) Fluent API
Fluent API使用 IsRowVersion() 方法来配置乐观并发。
为了测试Fluent API的配置,从 Students 类的 RowVersion 属性中删除 [Timestamp] 属性,并在 EducationContext 类中覆盖 ** onmodel()** 方法:
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Entity<Student>().Property(s => s.RowVersion).IsRowVersion();
base.OnModelCreating(modelBuilder);
}
配置非时间戳字段
如果不保留对并发检查的专用列,您仍然可以处理并发性。有些数据库不支持 RowVersion / Timestamp 类型的列。在这些类型的场景中,您可以使用Data Annotation或Fluent API配置来配置一个或多个用于并发检查的字段。
配置非时间戳字段
| Convention | None |
|---|---|
| Data Annotation | [ConcurrencyCheck] |
| Fluent API | .IsConcurrencyToken() |
1) Data Annotation
修改 Student 类,以使用 [ConcurrencyCheck] Data Annotation属性:
public class Student
{
public int StudentId { get; set; }
[ConcurrencyCheck]
public string RollNumber { get; set; }
public string FirstName { get; set; }
public string LastName { get; set; }
}
当应用程序运行时,代码首先创建 Students 表(参见下图)。数据库在 RollNumber 列中不为 [ConcurrencyCheck] 属性做任何特殊的事情。

但是当任何修改/更改发生在Students表时,代码首先处理并发检查。接下来阅读代码如何创建更新和删除并处理并发检查。
UPDATE SQL
exec sp_executesql N'UPDATE [dbo].[Students]
SET [RollNumber] = @0
WHERE (([StudentId] = @1) AND ([RollNumber] = @2))
',N'@0 nvarchar(max) ,@1 int,@2 nvarchar(max) ',@0=N'2',@1=1,@2=N'1'
注意 WHERE 条件。它在更新记录时比较 StudentId (主键)和 RollNumber 。
DELETE SQL
exec sp_executesql N'DELETE [dbo].[Students]
WHERE (([StudentId] = @0) AND ([RollNumber] = @1))',N'@0 int,@1 nvarchar(max) ',@0=1,@1=N'2'
在删除 Students 表的记录时,它还检查 StudentId 和 RollNumber 列值。如果 RollNumber 列值改变了,你现在正在更新那个记录,然后你就会得到OptimisticConcurrencyException。
2) Fluent API
使用Code First的 IsConcurrencyToken() 方法来处理非时间戳字段的并发性。
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Entity<Student>().Property(s => s.RollNumber).IsConcurrencyToken();
base.OnModelCreating(modelBuilder);
}
注意
为了测试并发效果,添加代码来更新Students表中已有的记录如下:
var student = context.Students.FirstOrDefault(u => u.StudentId == 1);
if (student != null)
{
student.RollNumber = "2";
context.SaveChanges();
}
在Visual Studio中为 context.SaveChanges() 行添加断点。在 SaveChanges() 方法执行之前,修改数据库中 Students 表记录条件是StudentId = 1。
UPDATE Students SET RollNumber = '123' WHERE StudentId = 1;
现在,如果你要执行下一行语句savechanges()然后你会得到如下的一个提示:
DbUpdateConcurrencyException:
Store update, insert, or delete statement affected an unexpected number of rows (0). Entities may have been modified or deleted since entities were loaded. Refresh ObjectStateManager entries.
DbUpdateConcurrencyException 异常抛出,因为记录已经修改为只读。
总结
在本文中,我们学习了如何通过Entity Framework Code First将专用字段保存在表中,或者通过添加特殊data annotation属性或Fluent API配置来配置处理乐观并发性的方法。
欢迎转载,转载请注明翻译原文出处(本文章),原文出处(原博客地址),然后谢谢观看
如果觉得我的翻译对您有帮助,请点击推荐支持:)
Entity Framework Code First实现乐观并发的更多相关文章
- Entity Framework Code First (三)Data Annotations
Entity Framework Code First 利用一种被称为约定(Conventions)优于配置(Configuration)的编程模式允许你使用自己的 domain classes 来表 ...
- Entity Framework Code First学习系列目录
Entity Framework Code First学习系列说明:开发环境为Visual Studio 2010 + Entity Framework 5.0+MS SQL Server 2012, ...
- Entity Framework Code First数据库连接
1. 安装Entity Framework 使用NuGet安装Entity Framework程序包:工具->库程序包管理器->程序包管理器控制台,执行以下语句: PM> Insta ...
- Entity Framework Code First属性映射约定
Entity Framework Code First与数据表之间的映射方式有两种实现:Data Annotation和Fluent API.本文中采用创建Product类为例来说明tity Fram ...
- Entity Framework Code First关系映射约定
本篇随笔目录: 1.外键列名默认约定 2.一对多关系 3.一对一关系 4.多对多关系 5.一对多自反关系 6.多对多自反关系 在关系数据库中,不同表之间往往不是全部都单独存在,而是相互存在关联的.两个 ...
- Entity Framework Code First执行SQL语句、视图及存储过程
1.Entity Framework Code First查询视图 Entity Framework Code First目前还没有特别针对View操作的方法,但对于可更新的视图,可以采用与Table ...
- Entity Framework Code First使用DbContext查询
DbContext.DbSet及DbQuery是Entity Framework Code First引入的3个新的类,其中DbContext用于保持数据库会话连接,实体变化跟踪及保存,DbSet用于 ...
- Entity Framework Code First添加修改及删除单独实体
对于一个单独实体的通常操作有3种:添加新的实体.修改实体以及删除实体. 1.添加新的实体 Entity Framework Code First添加新的实体通过调用DbSet.Add()方法来实现. ...
- Entity Framework Code First实体对象变动跟踪
Entity Framework Code First通过DbContext.ChangeTracker对实体对象的变动进行跟踪,实现跟踪的方式有两种:变动跟踪快照和变动跟踪代理. 变动跟踪快照:前面 ...
随机推荐
- 转载 java基础题(面试必知)
1.面向对象的特征有哪些方面 1.抽象:抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面.抽象并不打算了解全部问题,而只是选择其中的一部分,暂时不用部分细节.抽象包 ...
- eclipse没有联想功能的解决办法
1.我window->Preferences->Java->Editor->content assist 把 Enable auto activation 选项打上勾 :(如下 ...
- 使用SVG基本操作API
前面的话 本文将详细介绍SVG基本操作API,并使用这些API操作实例效果 基础API 在javascript中,可以使用一些基本的API来对SVG进行操作 [NS地址] 因为SVG定义在其自身的命令 ...
- Java学习记录:文件的输入输出流
Java中的输入.输出流中可以用于文件的读写,拷贝. 由于文件都是由字节组成的,可以将文件中的内容以字节的方式读取出来. 输入流还可以直接转换为图片来使用.其实ImageIcon提供了方法可以直接打开 ...
- PHP初入,简易网页整理(布局&特效的使用)
html><html> <head> <meta charset="UTF-8"> <title></title> ...
- Linux开机最简化
[root@localhost ~]# LANG=en [root@localhost ~]# for root in chkconfig --list|grep 3:on|awk '{print $ ...
- oop学习 计算器类的规划
类的学习 题目要求 采取面向对象的方法,四则运算自动出题软件,根据需求可以划分为几个类?每个类具有什么属性?每个类具有什么行为? 类与类之间是如何进行协作的?谁给谁发送消息?谁持有谁的引用? 该自动出 ...
- 【集美大学1411_助教博客】团队作业3——需求改进&系统设计 成绩
看到同学们越来越认真了,助教非常高兴.大家已经开始了alpha冲刺,请控制好进度.成功的关键就是不断迭代,不断迭代. 关于leangoo 我看到所有组的同学都已经开始使用leangoo,请大家把助教加 ...
- Swing-setBorder()用法-入门
注:本文内容转自:Swing编程边框(Border)的用法总结.内容根据笔者理解稍有整理. 函数说明: public void setBorder(Border border) 设置此组件的边框.Bo ...
- 201521123036 《Java程序设计》第4周学习总结
本周学习总结 1.1 尝试使用思维导图总结有关继承的知识点. 1.2 使用常规方法总结其他上课内容. 继承:extends子类将获得父类的属性与方法,并具有自身特有的属性与方法 抽取共同特征(行为与属 ...