分享一个 .NET 通过监听器拦截 EF 消息写日志的详细例子

前言

EF 开发效率确实很高也很便捷,但当它发生错误时,也挺让人头疼的,为什么?因为 EF 就像是一个黑盒子,一切全被封装起来,出错的时候很难定位原因,如果能够知道并打印 EF 生成的 SQL 语句,对于定位 EF 错误,就很有帮助。

程序员的才智是无限的,虽然 EF 有这个那个的问题,但程序员却总比这些问题多一个办法。

下面分享一个 .NET 通过监听器拦截 EF 消息写日志的详细例子。

Step By Step 步骤

  1. 创建自定义监听器类,拦截 EF 的命令执行事件(留意注释

    c# 复制代码
    /// <summary>
    /// 监听并写 EF 生成的 SQL 到日志
    /// </summary>
    public class EFIntercepterLogging : IDbCommandInterceptor
    {
        /// <summary>
        /// 实现接口的 NonQueryExecuting 方法
        /// </summary>
        /// <param name="command"></param>
        /// <param name="interceptionContext"></param>
        public void NonQueryExecuting(
            DbCommand command, 
            DbCommandInterceptionContext<int> interceptionContext)
        {
            LogWhenExecuting(command, interceptionContext);
        }
    
        /// <summary>
        /// 实现接口的 NonQueryExecuted 方法
        /// </summary>
        /// <param name="command"></param>
        /// <param name="interceptionContext"></param>
        public void NonQueryExecuted(
            DbCommand command, 
            DbCommandInterceptionContext<int> interceptionContext)
        {
            LogIfError(command, interceptionContext);
        }
    
        /// <summary>
        /// 实现接口的 ReaderExecuting 方法
        /// </summary>
        /// <param name="command"></param>
        /// <param name="interceptionContext"></param>
        public void ReaderExecuting(
            DbCommand command, 
            DbCommandInterceptionContext<DbDataReader> interceptionContext)
        {
            LogWhenExecuting(command, interceptionContext);
        }
    
        /// <summary>
        /// 实现接口的 ReaderExecuted 方法
        /// </summary>
        /// <param name="command"></param>
        /// <param name="interceptionContext"></param>
        public void ReaderExecuted(
            DbCommand command, 
            DbCommandInterceptionContext<DbDataReader> interceptionContext)
        {
            LogIfError(command, interceptionContext);
        }
    
        /// <summary>
        /// 实现接口的 ScalarExecuting 方法
        /// </summary>
        /// <param name="command"></param>
        /// <param name="interceptionContext"></param>
        public void ScalarExecuting(
            DbCommand command, 
            DbCommandInterceptionContext<object> interceptionContext)
        {
            LogWhenExecuting(command, interceptionContext);
        }
    
        /// <summary>
        /// 实现接口的 ScalarExecuted 方法
        /// </summary>
        /// <param name="command"></param>
        /// <param name="interceptionContext"></param>
        public void ScalarExecuted(
            DbCommand command, 
            DbCommandInterceptionContext<object> interceptionContext)
        {
            LogIfError(command, interceptionContext);
        }
    
        /// <summary>
        /// 写 EF 执行中的 SQL 日志,Debug 级别,用在 Executing 方法
        /// </summary>
        /// <typeparam name="TResult"></typeparam>
        /// <param name="command"></param>
        /// <param name="interceptionContext"></param>
        private void LogWhenExecuting<TResult>(
            DbCommand command, 
            DbCommandInterceptionContext<TResult> interceptionContext)
        {
            Logger.DebugFormat("Executing the following SQL: [{0}]", command.CommandText);
        }
    
        /// <summary>
        /// 出现异常时写日志到 Log4net
        /// </summary>
        /// <typeparam name="TResult"></typeparam>
        /// <param name="command"></param>
        /// <param name="interceptionContext"></param>
        private void LogIfError<TResult>(
            DbCommand command, 
            DbCommandInterceptionContext<TResult> interceptionContext)
        {
            if (interceptionContext.Exception != null)
            {
                var errMsg = new StringBuilder(16);
                errMsg.AppendLine("Error occurred when executing the following SQL: ");
                errMsg.AppendLine(command.CommandText);
                foreach (DbParameter param in command.Parameters)
                {
                    errMsg.AppendLine($"ParameterName:[{param.ParameterName}] -- DbType:[{param.DbType}] -- Value:[{param.Value}]");
                }
                Logger.Error(errMsg, interceptionContext.Exception);
            }
        }
    }
  2. 在 Global.asax 中的 Application_Start 方法注册监听器

    c# 复制代码
    protected void Application_Start()
    {
        AreaRegistration.RegisterAllAreas();
        FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
        RouteConfig.RegisterRoutes(RouteTable.Routes);
        BundleConfig.RegisterBundles(BundleTable.Bundles);
        //autofac注册
        AutofacConfig.RegisterConfig();
        //注册 EF log 监听器
        DbInterception.Add(new EFIntercepterLogging());
    }
  3. 这样就实现了监听器拦截 EF 消息写日志的功能,当运行程序执行 EF 语句时,就会自动将 EF 生成的 SQL 写到日志文件中

总结

通过实现自定义监听器类并注册监听器,可以跟踪和记录 EF 的操作和事件,在开发过程中更好地了解和调试 EF 的行为,不失为排查 EF 问题和优化 EF 性能的一个好方法。

我是老杨,一个奋斗在一线的资深研发老鸟,让我们一起聊聊技术,聊聊程序人生,共同学习,

相关推荐
future_studio33 分钟前
聊聊 Unity(小白专享、C# 小程序 之 图片播放器)
unity·小程序·c#
c#上位机6 小时前
wpf中Grid的MouseDown 事件无法触发的原因
c#·wpf
CodeCraft Studio8 小时前
国产化PDF处理控件Spire.PDF教程:如何在 C# 中从 HTML 和 PDF 模板生成 PDF
pdf·c#·html·.net·spire.pdf·pdf文档开发·html创建模板pdf
ysdysyn9 小时前
.NET 10深度解析:性能革新与开发生态的全新篇章
c#·.net
L X..12 小时前
Unity 光照贴图异常修复笔记
unity·c#·游戏引擎
reasonsummer13 小时前
【办公类-115-06】20250920职称资料上传04——docx复制、docx转PDF(课程表11个)
开发语言·windows·python·c#
fs哆哆1 天前
在VB.NET中,有没有 ?.这个运算符
java·开发语言·.net
荔园微风1 天前
微软ML.NET技术详解:从数据科学到生产部署的全栈解决方案
microsoft·.net
William_cl1 天前
一、前置基础(MVC学习前提)_核心特性_【C# 泛型入门】为什么说 List<T>是程序员的 “万能收纳盒“?避坑指南在此
学习·c#·mvc
c#上位机1 天前
wpf之命令
c#·wpf