当前位置: 首页 > news >正文

5-2EFCore性能优化

ASNoTracking

上文讲了实体类的跟踪以便执行SaveChanges操作。但是如果是查询操作,则实体类便不需要进行跟踪。

using TestDbContext ctx = new TestDbContext();
Book[] books = ctx.Books.AsNoTracking().Take(3).ToArray(); //设置不被跟踪
Book b1 = books[0];
b1.Title = "abc";//无效,因为不会被跟踪
EntityEntry entry1 = ctx.Entry(b1);

Find和FindAsync方法

当根据Id获取数据的时候,这两个方法会在上下文查找这个对象是否已经被跟踪,如果被跟踪,直接返回被跟踪的对象,不需要访问数据库。只有在本地没有找到这个对象时候,才去数据库查询,而Single方法则肯定要去访问数据库。

但是,在对象被跟踪之后,数据库中对应的数据被其他程序修改,如果使用Find方法则可能返回旧数据

Book b = ctx.Books.Find(2)

全局查询筛选器

设置全局查询筛选器,EF Core会自动将全局查询筛选器应用于涉及这个实体类的所有LINQ查询。常应用“软删除”功能,即并不真的删除数据,而是增加某列指示该数据是否被删除。

例如在Book类中增加一个bool属性值IsDeleted,标记是否被删除

在配置类中增加builder.HasQueryFilter(b=>b.IsDeleted==false),这样在对Book实体类的查询都会自动加上b.IsDeleted==false这个筛选器

如果需要查询被删除的数据,可以使用IgnoreQueryFilters来临时忽略过滤器

ctx.Books.IgnoreQueryFilters().Where(b=>b.Title.Contains("a"))

注意:如果启动了全局查询筛选器,会导致全表扫描,性能降低。

悲观并发控制

为了避免多个用户同时操作资源造成并发冲突问题,通常要进行并发控制。

悲观并发控制一般采用行锁、表锁等排他锁对资源进行锁定。

要使用悲观并发控制需要自行编写SQL语句。

案例:抢房子

class House
{public long Id { get; set; }public string Name { get; set; }public string? Owner { get; set; }
}using Microsoft.EntityFrameworkCore;Console.WriteLine("请输入您的姓名");
string name = Console.ReadLine();
using MyDbContext ctx = new MyDbContext();
//锁和事务是相关的,使用BeginTransactionAsync创建一个事务,并在后面提交事务
using var tx = await ctx.Database.BeginTransactionAsync();
//for update创建用于更新的锁,如果其他用户也使用for update查询id=1的数据,则查询被挂起,这是mysql的语法
var h1 = await ctx.Houses.FromSqlInterpolated($"select * from T_Houses where Id=1 for update").SingleAsync();if (string.IsNullOrEmpty(h1.Owner))
{await Task.Delay(5000);h1.Owner = name;await ctx.SaveChangesAsync();Console.WriteLine("抢到手了");
}
else
{if (h1.Owner == name){Console.WriteLine("这个房子已经是你的了,不用抢");}else{Console.WriteLine($"这个房子已经被{h1.Owner}抢走了");}
}
await tx.CommitAsync();//提交事务

悲观并发控制使用简单,使用排它锁就可以,但是如果并发量大则严重影响使用性能。

乐观并发控制

EF Core内置了使用并发令牌列实现乐观并发控制,并发令牌列通常就是被并发操作影响的列,比如上面的house类,其中owner属性就可以作为并发令牌列。执行update hourse set Owner=新值 where Id=1 and Owner = 旧值,如果其他人更改了Owner则where语句就是false,此时SaveChanges方法会抛出DbUpdateConcurrencyException

只需要在配置类中做设置

class HouseConfig : IEntityTypeConfiguration<House>
{public void Configure(EntityTypeBuilder<House> builder){builder.ToTable("T_Houses");//在配置类中使用IsConcurrencyToken把Owner列设置为并发并令牌属性builder.Property(h => h.Owner).IsConcurrencyToken();}
}
using MyDbContext ctx = new MyDbContext();
var h1 = await ctx.Houses.SingleAsync(h => h.Id == 1);
if (string.IsNullOrEmpty(h1.Owner))
{await Task.Delay(5000);h1.Owner = name;try{await ctx.SaveChangesAsync();Console.WriteLine("抢到手了");}catch (DbUpdateConcurrencyException ex){//通过DbUpdateConcurrencyException类的Entries属性获取并发修改冲突的EntityEntry//并通过EntityEntry类中的GetDatabaseValuesAsync获取当前数据库的值var entry = ex.Entries.First();var dbValues = await entry.GetDatabaseValuesAsync();string newOwner = dbValues.GetValue<string>(nameof(House.Owner));Console.WriteLine($"并发冲突,被{newOwner}提前抢走了");}
}
else{...}

有时候,无法确定到底那个属性适合作为并发令牌,这种情况下,可以设置一个额外的并发令牌属性,例如使用GUID


文章转载自:

http://h0PLmMqc.zxwbL.cn
http://AwsrAz1N.zxwbL.cn
http://FDg4JTYz.zxwbL.cn
http://wPv2eAos.zxwbL.cn
http://UZ6s3REr.zxwbL.cn
http://VNC6UcfE.zxwbL.cn
http://ZYa66NIf.zxwbL.cn
http://IyAtzTD2.zxwbL.cn
http://9K5qp6xH.zxwbL.cn
http://LB97CJBG.zxwbL.cn
http://IKmzY6NM.zxwbL.cn
http://yO2V78R0.zxwbL.cn
http://XhHxROqO.zxwbL.cn
http://nCPpLxRJ.zxwbL.cn
http://qn5AqHsj.zxwbL.cn
http://Qxe1dhzk.zxwbL.cn
http://dod1xh1g.zxwbL.cn
http://aQTxlbyI.zxwbL.cn
http://A0rXodms.zxwbL.cn
http://zIFRx0GB.zxwbL.cn
http://nnhYZtH6.zxwbL.cn
http://9l2FFupJ.zxwbL.cn
http://pXJxt7tG.zxwbL.cn
http://ngdMQNC3.zxwbL.cn
http://oD7h24xB.zxwbL.cn
http://jtq3qthc.zxwbL.cn
http://nLi6gi6s.zxwbL.cn
http://OXKbHmw5.zxwbL.cn
http://21T34NuA.zxwbL.cn
http://aSH5zcFA.zxwbL.cn
http://www.dtcms.com/a/371378.html

相关文章:

  • 《sklearn机器学习——管道和复合估计器》联合特征(FeatureUnion)
  • 大数据毕业设计选题推荐-基于大数据的贵州茅台股票数据分析系统-Spark-Hadoop-Bigdata
  • C++ 容器——unordered_xxx
  • 常见设计模式详解
  • 网络爬虫(web crawler)
  • 319章:使用Scrapy框架构建分布式爬虫
  • (LeetCode 面试经典 150 题) 67. 二进制求和(位运算、字符串)
  • 【linux kernel 常用数据结构和设计模式】【数据结构 3】【模拟input子系统input_dev和input_handler之间的多对多关系】
  • 怎么用CXL加速数据库?· SIGMOD‘25
  • Day23_【机器学习—聚类算法—K-Means聚类 及评估指标SSE、SC、CH】
  • Uniapp 怎么修改项目图标和启动页
  • macos安装openjdk17
  • 像素图生成小程序开发全解析:从图片上传到Excel图纸
  • 运维服务方案,运维巡检方案,运维安全保障方案文件
  • 计算机网络相关
  • 【代码讲解】北通手柄遥控 + Mujoco 仿真 SO-ARM100 机械臂末端位姿
  • ARM 体系结构
  • 云平台面试内容(一)
  • React Hooks 钩子
  • js逆向之JSEncrypt的加密
  • Balabolka:免费高效的文字转语音软件
  • 第3天-Jenkins详解-3
  • Linux内核进程管理子系统有什么第四十三回 —— 进程主结构详解(39)
  • Apache 2.0 开源协议详解:自由、责任与商业化的完美平衡-优雅草卓伊凡
  • 【iOS】多界面传值
  • 【C++】简单介绍lambda表达式
  • uv 包管理器:优势解析与使用指南
  • Android studio的adb和终端的adb互相抢占端口
  • 同类软件对比(四):Jupyter vs PyCharm vs VS Code:Python开发工具终极选择指南
  • 【MySQL】数据库的基本操作