EF Core3.0+ 通过拦截器实现读写分离与SQL日志记录
本文主要是讲解EF Core3.0+ 通过拦截器实现读写分离与SQL日志记录
注意拦截器只有EF Core3.0+ 支持,2.1请考虑上下文工厂的形式实现.
说点题外话..
一晃又大半年没更新技术博客..唉,去年一年发生了太多事情..博主真的 一言难尽..
有兴趣的可以去看看:记录一下,也许是转折,也许是结束,也许是新希望的一年
1.通过拦截器实现读写分离
先讲一下本文实现的方式吧
SQL 通过数据库本身的功能 实现主从备份 大概原理如图:


EF Core在查询的时候通过DbCommandInterceptor 拦截器(PS:这个功能在EF6.0+中也实现了)来拦截对数据库的访问,从而切换主从数据库
下面直接上代码吧
首先我们创建一个类 继承DbCommandInterceptor:
public class DbMasterSlaveCommandInterceptor : DbCommandInterceptor
{
private string _masterConnectionString;
private string _slaveConnectionString; public DbMasterSlaveCommandInterceptor(string masterConnectionString, string slaveConnectionString)
{
_masterConnectionString = masterConnectionString;
_slaveConnectionString = slaveConnectionString;
}
}
通过构造函数传递主库连接地址与从库地址(可有多个 通过"|"分割)
添加一个随机分配从表读取连接的方法(PS:这里只是demo所以很简陋的随机,如果正式要用,应包含权重判断,定时心跳从库连接情况,请自行修改):
/// <summary>
/// 通过随机数分配获取多个从库
/// </summary>
/// <returns></returns>
private string GetSlaveConnectionString()
{
var readArr = _slaveConnectionString.Split(new char[] { '|' }, StringSplitOptions.RemoveEmptyEntries);
var resultConn = string.Empty;
if (readArr != null && readArr.Any())
{
resultConn = readArr[Convert.ToInt32(Math.Floor((double)new Random().Next(0, readArr.Length)))];
}
return resultConn;
}
添加判断是否主从操作连接方法:
private void UpdateToSlave(DbCommand command)
{
//判断是否配置了主从分离
if (!string.IsNullOrWhiteSpace(GetSlaveConnectionString()))//如果配置了读写分离,就进入判断
{
//判断是否为插入语句(EF 插入语句会通过Reader执行并查询主键),否则进入
if (command.CommandText.ToLower().StartsWith("insert", StringComparison.InvariantCultureIgnoreCase) == false)
{
// 判断当前会话是否处于分布式事务中
bool isDistributedTran = Transaction.Current != null &&
Transaction.Current.TransactionInformation.Status !=
TransactionStatus.Committed;
//判断该 context 是否处于普通数据库事务中
bool isDbTran = command.Transaction != null;
//如果不处于事务中,则执行从服务器查询
if (!isDbTran && !isDistributedTran)
{
command.Connection.Close();
command.Connection.ConnectionString = GetSlaveConnectionString();
command.Connection.Open(); } }
}
}
重载DbCommandInterceptor当中的拦截方法,代码如下:
//如果是写入,则正常执行
public override InterceptionResult<int> NonQueryExecuting(DbCommand command, CommandEventData eventData, InterceptionResult<int> result)
{
return base.NonQueryExecuting(command, eventData, result);
}
public override Task<InterceptionResult<int>> NonQueryExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<int> result, CancellationToken cancellationToken = default)
{
return base.NonQueryExecutingAsync(command, eventData, result, cancellationToken);
} public override InterceptionResult<DbDataReader> ReaderExecuting(DbCommand command, CommandEventData eventData, InterceptionResult<DbDataReader> result)
{
this.UpdateToSlave(command);
return base.ReaderExecuting(command, eventData, result);
} public override Task<InterceptionResult<DbDataReader>> ReaderExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<DbDataReader> result, CancellationToken cancellationToken = default)
{
this.UpdateToSlave(command);
return base.ReaderExecutingAsync(command, eventData, result, cancellationToken);
} public override InterceptionResult<object> ScalarExecuting(DbCommand command, CommandEventData eventData, InterceptionResult<object> result)
{
this.UpdateToSlave(command);
return base.ScalarExecuting(command, eventData, result);
} public override Task<InterceptionResult<object>> ScalarExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<object> result, CancellationToken cancellationToken = default)
{
this.UpdateToSlave(command);
return base.ScalarExecutingAsync(command, eventData, result, cancellationToken);
}
最后在EF core的上下文中注入拦截器(PS:我这里使用的Autofac模块注入):
builder.Register(
c =>
{
var optionsBuilder = new DbContextOptionsBuilder<TestEFContext>();
//注入拦截器
optionsBuilder.AddInterceptors(new DbMasterSlaveCommandInterceptor(WriteConnect, ReadConnect));
//MaxBatchSize 处理批量操作BUG
optionsBuilder.UseMysql(WriteConnect, b=>b.MaxBatchSize(1));
return optionsBuilder.Options;
}
).As<DbContextOptions<TestEFContex>>().SingleInstance();
这样就实现了通过拦截器实现读写分离.
2.通过拦截器实现SQL日志记录
同理,我们可以通过拦截器实现EF Core SQL语句的记录与调试
首先我们创建一个新的拦截器DBlogCommandInterceptor 如下:
public class DBlogCommandInterceptor : DbCommandInterceptor
{
//创建一个队列记录SQL执行时间
static readonly ConcurrentDictionary<DbCommand, DateTime> MStartTime = new ConcurrentDictionary<DbCommand, DateTime>();
private ILogger<DBlogCommandInterceptor> _logger { get; set; }
//通过构造函数注入日志
public DBlogCommandInterceptor(ILogger<DBlogCommandInterceptor> Logger)
{
_logger = Logger;
}
}
创建2个私有的方法,一个记录执行开始时间,一个记录SQL
//记录SQL开始执行的时间
private void OnStart(DbCommand command)
{
MStartTime.TryAdd(command, DateTime.Now);
}
//通过_logger输出日志
private void Log(DbCommand command)
{ DateTime startTime;
TimeSpan duration;
//得到此command的开始时间
MStartTime.TryRemove(command, out startTime);
if (startTime != default(DateTime))
{
duration = DateTime.Now - startTime;
}
else
{
duration = TimeSpan.Zero;
}
var parameters = new StringBuilder();
//循环获取执行语句的参数值
foreach (DbParameter param in command.Parameters)
{
parameters.AppendLine(param.ParameterName + " " + param.DbType + " = " + param.Value);
}
_logger.LogInformation("{starttime}开始执行SQL语句:{sql},参数:{canshu},执行时间{readtime}",
startTime.ToString(), command.CommandText, parameters.ToString(), duration.TotalSeconds); }
最后重载拦截器的方法:
public override InterceptionResult<int> NonQueryExecuting(DbCommand command, CommandEventData eventData, InterceptionResult<int> result)
{
OnStart(command);
return base.NonQueryExecuting(command, eventData, result);
}
public override Task<InterceptionResult<int>> NonQueryExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<int> result, CancellationToken cancellationToken = default)
{
OnStart(command);
return base.NonQueryExecutingAsync(command, eventData, result, cancellationToken);
}
public override int NonQueryExecuted(DbCommand command, CommandExecutedEventData eventData, int result)
{
Log(command);
return base.NonQueryExecuted(command, eventData, result);
}
public override Task<int> NonQueryExecutedAsync(DbCommand command, CommandExecutedEventData eventData, int result, CancellationToken cancellationToken = default)
{
Log(command);
return base.NonQueryExecutedAsync(command, eventData, result, cancellationToken);
} public override InterceptionResult<object> ScalarExecuting(DbCommand command, CommandEventData eventData, InterceptionResult<object> result)
{
OnStart(command);
return base.ScalarExecuting(command, eventData, result);
}
public override Task<InterceptionResult<object>> ScalarExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<object> result, CancellationToken cancellationToken = default)
{
OnStart(command);
return base.ScalarExecutingAsync(command, eventData, result, cancellationToken);
}
public override object ScalarExecuted(DbCommand command, CommandExecutedEventData eventData, object result)
{
Log(command);
return base.ScalarExecuted(command, eventData, result);
}
public override Task<object> ScalarExecutedAsync(DbCommand command, CommandExecutedEventData eventData, object result, CancellationToken cancellationToken = default)
{
Log(command);
return base.ScalarExecutedAsync(command, eventData, result, cancellationToken);
} public override InterceptionResult<DbDataReader> ReaderExecuting(DbCommand command, CommandEventData eventData, InterceptionResult<DbDataReader> result)
{
OnStart(command);
return base.ReaderExecuting(command, eventData, result);
}
public override Task<InterceptionResult<DbDataReader>> ReaderExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<DbDataReader> result, CancellationToken cancellationToken = default)
{
OnStart(command);
return base.ReaderExecutingAsync(command, eventData, result, cancellationToken);
} public override Task<DbDataReader> ReaderExecutedAsync(DbCommand command, CommandExecutedEventData eventData, DbDataReader result, CancellationToken cancellationToken = default)
{
Log(command);
return base.ReaderExecutedAsync(command, eventData, result, cancellationToken);
}
public override DbDataReader ReaderExecuted(DbCommand command, CommandExecutedEventData eventData, DbDataReader result)
{
Log(command);
return base.ReaderExecuted(command, eventData, result);
}
这样,我们就实现了通过拦截器实现SQL日志记录~效果如下:

调试SQL语句就方便了很多~
EF Core3.0+ 通过拦截器实现读写分离与SQL日志记录的更多相关文章
- ASP.NET Core 3.0 gRPC 拦截器
目录 ASP.NET Core 3.0 使用gRPC ASP.NET Core 3.0 gRPC 双向流 ASP.NET Core 3.0 gRPC 拦截器 一. 前言 前面两篇文章给大家介绍了使用g ...
- AOP框架Dora.Interception 3.0 [3]: 拦截器设计
对于所有的AOP框架来说,多个拦截器最终会应用到某个方法上.这些拦截器按照指定的顺序构成一个管道,管道的另一端就是针对目标方法的调用.从设计角度来将,拦截器和中间件本质是一样的,那么我们可以按照类似的 ...
- Spring Boot2.0 设置拦截器
所有功能完成 配置登录认证 配置拦截器 在spring boot2.0 之后 通过继承这个WebMvcConfigurer类 就可以完成拦截 新建包com.example.interceptor; 创 ...
- springboot 2.0+ 自定义拦截器
之前项目的springboot自定义拦截器使用的是继承WebMvcConfigurerAdapter重写常用方法的方式来实现的. 以下WebMvcConfigurerAdapter 比较常用的重写接口 ...
- SpringMVC案例3----spring3.0项目拦截器、ajax、文件上传应用
依然是项目结构图和所需jar包图: watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvYmVuamFtaW5fd2h4/font/5a6L5L2T/fontsi ...
- springboot2.0+ 使用拦截器导致静态资源被拦截
在spring1.0+的版本中,配置拦截器后是不会拦截静态资源的.其配置如下: @Configuration public class WebMvcConfig extends WebMvcConfi ...
- EF通用数据层封装类(支持读写分离,一主多从)
浅谈orm 记得四年前在学校第一次接触到 Ling to Sql,那时候瞬间发现不用手写sql语句是多么的方便,后面慢慢的接触了许多orm框架,像 EF,Dapper,Hibernate,Servic ...
- EF多租户实例:演变为读写分离
前言 我又来写关于多租户的内容了,这个系列真够漫长的. 如无意外这篇随笔是最后一篇了.内容是讲关于如何利用我们的多租户库简单实现读写分离. 分析 对于读写分离,其实有很多种实现方式,但是总体可以分以下 ...
- Akka-CQRS(0)- 基于akka-cluster的读写分离框架,构建gRPC移动应用后端架构
上一篇我们讨论了akka-cluster的分片(sharding)技术.在提供的例子中感觉到akka这样的分布式系统工具特别适合支持大量的带有内置状态的,相对独立完整的程序在集群节点上分布运算.这里重 ...
随机推荐
- Linux 驱动框架---cdev字符设备驱动和misc杂项设备驱动
字符设备 Linux中设备常见分类是字符设备,块设备.网络设备,其中字符设备也是Linux驱动中最常用的设备类型.因此开发Linux设备驱动肯定是要先学习一下字符设备的抽象的.在内核中使用struct ...
- React Hooks 内部实现原理
React Hooks 内部实现原理 源码分析 // 链表 React Hooks 原理剖析 refs https://reactjs.org/docs/hooks-intro.html https: ...
- js binary search algorithm
js binary search algorithm js 二分查找算法 二分查找, 前置条件 存储在数组中 有序排列 理想条件: 数组是递增排列,数组中的元素互不相同; 重排 & 去重 顺序 ...
- 比特币市场活跃,VAST发行在即!
截至1月25日13:30,BTC合约多空持仓人数比为1.44,市场做多人数占据优势:季度合约基差保持在1255美元上方,永续合约资金费率为正,交割及永续合约持仓总量为19.5亿美元,总体上多军占优:B ...
- SSL/TLS协议详解(下)——TLS握手协议
本文转载自SSL/TLS协议详解(下)--TLS握手协议 导语 在博客系列的第2部分中,对证书颁发机构进行了深入的讨论.在这篇文章中,将会探索整个SSL/TLS握手过程,在此之前,先简述下最后这块内容 ...
- .net实现filestream类复制文件
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.I ...
- 基于tcp的应用层消息边界如何定义
聊聊基于tcp的应用层消息边界如何定义 背景 2018年笔者有幸接触一个项目要用到长连接实现云端到设备端消息推送,所以借机了解过相关的内容,最终是通过rabbitmq+mqtt实现了相关功能,同时在心 ...
- day1 分布式基础概念
1. 分布式:一个业务分拆多个子业务,部署在不同的服务器上集群:同一个业务,部署在多个服务器上节点:集群中的一个服务器 2.远程调用 分布式系统中调用其它主机 springcloud用http+jso ...
- 【Notes_1】现代图形学入门——计算机图形学概述
跟着闫令琪老师的课程学习,总结自己学习到的知识点 课程网址GAMES101 B站课程地址GAMES101 课程资料百度网盘[提取码:0000] 计算机图形学概述 计算机图形学是一门将模型转化到屏幕上图 ...
- ElasticSearch7.9.2设置密码
1:设置ElasticSearch的密码 1.1:停止运行ElasticSearch,修改配置. vim elasticsearch-7.9.2/config/elasticsearch.yml 新增 ...