EntityFrameworkCore中的事务

实验目示

观察EFCore中使用Transaction或不使用的情况下,数据的一致性情况。

实验环境

开发环境

  • Windows 10
  • VS2017
  • Mssql 2014

组件

  • Microsoft.EntityFrameworkCore 2.2.0
  • Microsoft.EntityFrameworkCore.SqlServer 2.2.0
  • System.Data.SqlClient 4.6.1

实验记录

实验数据库

建库

USE [master]
GO

/****** Object:  Database [Basket]    Script Date: 2019-08-13 11:01:45 ******/
CREATE DATABASE [Basket]
 CONTAINMENT = NONE
 ON  PRIMARY 
( NAME = N'Basket', FILENAME = N'D:\Database\MSSQL12.MSSQLSERVER\MSSQL\DATA\Basket.mdf' , SIZE = 376832KB , MAXSIZE = UNLIMITED, FILEGROWTH = 1024KB )
 LOG ON 
( NAME = N'Basket_log', FILENAME = N'D:\Database\MSSQL12.MSSQLSERVER\MSSQL\DATA\Basket_log.ldf' , SIZE = 470144KB , MAXSIZE = 2048GB , FILEGROWTH = 10%)
GO

ALTER DATABASE [Basket] SET COMPATIBILITY_LEVEL = 120
GO

IF (1 = FULLTEXTSERVICEPROPERTY('IsFullTextInstalled'))
begin
EXEC [Basket].[dbo].[sp_fulltext_database] @action = 'enable'
end
GO

ALTER DATABASE [Basket] SET ANSI_NULL_DEFAULT OFF 
GO

ALTER DATABASE [Basket] SET ANSI_NULLS OFF 
GO

ALTER DATABASE [Basket] SET ANSI_PADDING OFF 
GO

ALTER DATABASE [Basket] SET ANSI_WARNINGS OFF 
GO

ALTER DATABASE [Basket] SET ARITHABORT OFF 
GO

ALTER DATABASE [Basket] SET AUTO_CLOSE OFF 
GO

ALTER DATABASE [Basket] SET AUTO_SHRINK OFF 
GO

ALTER DATABASE [Basket] SET AUTO_UPDATE_STATISTICS ON 
GO

ALTER DATABASE [Basket] SET CURSOR_CLOSE_ON_COMMIT OFF 
GO

ALTER DATABASE [Basket] SET CURSOR_DEFAULT  GLOBAL 
GO

ALTER DATABASE [Basket] SET CONCAT_NULL_YIELDS_NULL OFF 
GO

ALTER DATABASE [Basket] SET NUMERIC_ROUNDABORT OFF 
GO

ALTER DATABASE [Basket] SET QUOTED_IDENTIFIER OFF 
GO

ALTER DATABASE [Basket] SET RECURSIVE_TRIGGERS OFF 
GO

ALTER DATABASE [Basket] SET  DISABLE_BROKER 
GO

ALTER DATABASE [Basket] SET AUTO_UPDATE_STATISTICS_ASYNC OFF 
GO

ALTER DATABASE [Basket] SET DATE_CORRELATION_OPTIMIZATION OFF 
GO

ALTER DATABASE [Basket] SET TRUSTWORTHY OFF 
GO

ALTER DATABASE [Basket] SET ALLOW_SNAPSHOT_ISOLATION OFF 
GO

ALTER DATABASE [Basket] SET PARAMETERIZATION SIMPLE 
GO

ALTER DATABASE [Basket] SET READ_COMMITTED_SNAPSHOT OFF 
GO

ALTER DATABASE [Basket] SET HONOR_BROKER_PRIORITY OFF 
GO

ALTER DATABASE [Basket] SET RECOVERY FULL 
GO

ALTER DATABASE [Basket] SET  MULTI_USER 
GO

ALTER DATABASE [Basket] SET PAGE_VERIFY CHECKSUM  
GO

ALTER DATABASE [Basket] SET DB_CHAINING OFF 
GO

ALTER DATABASE [Basket] SET FILESTREAM( NON_TRANSACTED_ACCESS = OFF ) 
GO

ALTER DATABASE [Basket] SET TARGET_RECOVERY_TIME = 0 SECONDS 
GO

ALTER DATABASE [Basket] SET DELAYED_DURABILITY = DISABLED 
GO

ALTER DATABASE [Basket] SET  READ_WRITE 
GO

建表

USE [Basket]
GO
/****** Object:  Table [dbo].[UserClasses]    Script Date: 2019-08-13 11:01:37 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
SET ANSI_PADDING ON
GO
CREATE TABLE [dbo].[UserClasses](
    [Id] [int] IDENTITY(1,1) NOT NULL,
    [UserId] [int] NULL,
    [CaseId] [varchar](50) NULL,
    [Name] [varchar](50) NULL
) ON [PRIMARY]

GO
SET ANSI_PADDING OFF
GO
/****** Object:  Table [dbo].[Users]    Script Date: 2019-08-13 11:01:37 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
SET ANSI_PADDING ON
GO
CREATE TABLE [dbo].[Users](
    [Id] [int] IDENTITY(1,1) NOT NULL,
    [CaseId] [varchar](50) NULL,
    [Name] [varchar](50) NULL
) ON [PRIMARY]

GO
SET ANSI_PADDING OFF
GO

实验模型

 public class AwpGroundDbContext : DbContext
    {
        public AwpGroundDbContext()
        {

        }
        public virtual DbSet<User> Users { get; set; }
        public virtual DbSet<UserClass> UserClasses { get; set; }
        protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
        {
            optionsBuilder.UseSqlServer("Server=.;Database=Basket;Trusted_Connection=True;");
        }

    }
    public class User
    {
        public int Id { get; set; }
        public string CaseId { get; set; }
        public string Name { get; set; }
    }
    public class UserClass
    {
        [Key]
        public int Id { get; set; }
        public int UserId { get; set; }
        public string CaseId { get; set; }
        public string Name { get; set; }
    }

实验过程

1、不使用事务,对两个表的数据分别SaveChanges(),并在中间抛出异常

            using(var db = new AwpGroundDbContext())
            {
                var user = new User
                {
                    Id = 0,
                    CaseId = Guid.NewGuid().ToString("N"),
                    Name = "user3"
                };
                db.Users.Add(user);
                db.SaveChanges();
                throw new Exception("在这里抛出了异常");
                db.UserClasses.Add(
                    new UserClass
                    {
                        Id = 0,
                        UserId = user.Id,
                        CaseId = user.CaseId,
                        Name = "小一班"
                    });

                db.SaveChanges();

            }

数据表Users成功保存了数据,UserClasses无数据。

Users表
UserClasses

2、一次性SaveChanges(),未发生异常的情况下,是否能够拿到UserId呢?

            using(var db = new AwpGroundDbContext())
            {
                var user = new User
                {
                    Id = 0,
                    CaseId = Guid.NewGuid().ToString("N"),
                    Name = "user3"
                };
                db.Users.Add(user);
                db.UserClasses.Add(
                    new UserClass
                    {
                        Id = 0,
                        UserId = user.Id,
                        CaseId = user.CaseId,
                        Name = "小一班"
                    });

                db.SaveChanges();

            }

答案是,没有!所以,主从结构的保存时,不适用这种方式

无法获取UserId

3、使用事务,分别调用SaveChanges(),能否拿到UserId呢?

            using (var db = new AwpGroundDbContext())
            {
                using (var tran = db.Database.BeginTransaction())
                {
                    var user = new User
                    {
                        Id = 0,
                        CaseId = Guid.NewGuid().ToString("N"),
                        Name = "user3"
                    };
                    db.Users.Add(user);
                    db.SaveChanges();
                    db.UserClasses.Add(
                        new UserClass
                        {
                            Id = 0,
                            UserId = user.Id,
                            CaseId = user.CaseId,
                            Name = "小一班"
                        });

                    db.SaveChanges();
                    tran.Commit();
                }

            }

显然这样是可以的。

拿到UserId

4、使用事务,中间添加异常,未调用Rollback(),是否会出现保存了部分数据的情况呢?

            using (var db = new AwpGroundDbContext())
            {
                using (var tran = db.Database.BeginTransaction())
                {
                    var user = new User
                    {
                        Id = 0,
                        CaseId = Guid.NewGuid().ToString("N"),
                        Name = "user5"
                    };
                    db.Users.Add(user);
                    db.SaveChanges();
                    throw new Exception("出错了");
                    db.UserClasses.Add(
                        new UserClass
                        {
                            Id = 0,
                            UserId = user.Id,
                            CaseId = user.CaseId,
                            Name = "小一班"
                        });

                    db.SaveChanges();
                    tran.Commit();
                }

            }

可以看到,User5并未保存。

未保存

结论

由于SaveChanges()单独创建了一个事务执行,所以当我们要保证多个表的数据一致性时,需要创建一个事务,并在事务中执行SaveChanges(),当其中出现错误时,事务未提交,不影响数据库,所以不需要手工Rollback。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容