C# 实现调用函数,打印日志(通过反射代理、非IOC)

在这里插入图片描述

🎈个人主页:靓仔很忙i
💻B 站主页:👉B站👈
🎉欢迎 👍点赞✍评论⭐收藏
🤗收录专栏:C#
🤝希望本文对您有所裨益,如有不足之处,欢迎在评论区提出指正,让我们共同学习、交流进步!


需求:对于不方便debug的程序,查看日志是一个特别好的调试方式,但是在.net framework中,想实现面向切面打印日志特别困难。有人可能会想到使用Castle,使用Castle确实ok。但它需要依赖注入,需要框架去托管所有需要AOP的对象,系统中还有很多静态类,想实现这些都类的日志打印,我还需要去包装类库中的所有的静态类。这显然不现实,本文就是为了解决这个问题而写。

解决方案:为此,笔者写了一个Logger类,提供了CallStatic,CallMethod方式分别去调用方法,调用方法的时候,打印日志。调用思路也很简单,直接调用就行,需要调用的方法,通过参数传递即可。废话不多说,直接上干货。

  • 程序需要三个文件
    在这里插入图片描述
    • Logger:日志类
    • MyService: 测试使用的方法
    • Program: 入口
  • Logger文件
    using System;using System.Linq;using System.Linq.Expressions;using System.Reflection;namespace AutoTemplate.log{public static class Logger{public static void CallStatic(Expression<Action> expression){var methodInfo = (MethodCallExpression)expression.Body;var method = GetMethodInfo(methodInfo);var parameters = GetParameters(methodInfo);PrintStartLog(method, parameters);method.Invoke(null, parameters);PrintEndLog(method);}public static T CallStatic<T>(Expression<Func<T>> expression){var methodInfo = (MethodCallExpression)expression.Body;var method = GetMethodInfo(methodInfo);var parameters = GetParameters(methodInfo);PrintStartLog(method, parameters);var result = (T)method.Invoke(null, parameters);PrintEndLog(method);return result;}public static void CallMethod<T>(this T instance, Expression<Action<T>> expression){var methodInfo = (MethodCallExpression)expression.Body;var method = GetMethodInfo(methodInfo);var parameters = GetParameters(methodInfo);PrintStartLog(method, parameters);method.Invoke(instance, parameters);PrintEndLog(method);}public static TResult CallMethod<T, TResult>(this T instance, Expression<Func<T, TResult>> expression){var methodInfo = (MethodCallExpression)expression.Body;var method = GetMethodInfo(methodInfo);var parameters = GetParameters(methodInfo);PrintStartLog(method, parameters);var result = (TResult)method.Invoke(instance, parameters);PrintEndLog(method);return result;}private static MethodInfo GetMethodInfo(MethodCallExpression methodCall){return methodCall.Method;}private static object[] GetParameters(MethodCallExpression methodCall){object[] parameters = methodCall.Arguments.Count > 0? methodCall.Arguments.Select(arg => Expression.Lambda(arg).Compile().DynamicInvoke()).ToArray(): Array.Empty<object>();return parameters;}private static void PrintStartLog(MethodInfo method, object[] parameters) {Console.WriteLine($"==========================================");Console.WriteLine($"[Log Start]开始调用方法: {method.Name}({string.Join(", ", parameters)})");}private static void PrintEndLog(MethodInfo method){Console.WriteLine($"[Log End]方法调用成功: {method.Name}");}}}
  • MyService
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace AutoTemplate
{public class MyService{public static void NoArgNoResultStaticMethod(){Console.WriteLine("静态方法:无参无返回值");}public static void OneArgStaticMethod(string a){Console.WriteLine($"静态方法:一个参数无返回值,参数:{a}");}public static string ResultStaticMethod(){Console.WriteLine($"静态方法:无参数有返回值,返回值:hello");return "这是ResultStaticMethod返回值";}public void NoArgNoResultMethod(){Console.WriteLine("方法:无参无返回值");}public void OneArgMethod(string aaa){Console.WriteLine($"方法:一个参数无返回值,参数:{aaa}");}public int ReturnMethod(){Console.WriteLine($"方法:无参一个返回值");return 0;}public string OneArgReturnMethod(string e){Console.WriteLine($"方法:一个参数一个返回值,参数:{e}");return "这是OneArgReturnMethod返回值";}public string ThreeArgOneResultMethod(string e,float f,int g){Console.WriteLine($"方法:三个参数一个返回值,参数:{e},{f},{g}");return "这是ThreeArgOneResultMethod返回值";}}
}
  • Program
using AutoTemplate.log;
using FlaUI.UIA3;
using PATool.Core.Automation;
using System;
using System.Diagnostics;
using System.Linq;
using System.Threading;namespace AutoTemplate
{internal class Program{static void Main(string[] args){//静态方法调用Logger.CallStatic(() => MyService.NoArgNoResultStaticMethod());Logger.CallStatic(() => MyService.OneArgStaticMethod("abc"));var staticRes= Logger.CallStatic(() => MyService.ResultStaticMethod());var processRes = Logger.CallStatic(() => Process.Start("notepad++"));//对象方法调用var myService = new MyService();myService.CallMethod(s => s.NoArgNoResultMethod());myService.CallMethod(s => s.OneArgMethod("hello"));var res1=myService.CallMethod(s => s.ReturnMethod());var res2=myService.CallMethod(s => s.OneArgReturnMethod("xlwang"));var res3 = myService.CallMethod(s => s.ThreeArgOneResultMethod("xlwang",1.2f,5));}}
}
  • 运行结果
    在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/447073.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

大数据ETL数据提取转换和加载处理

什么是 ETL&#xff1f; 提取转换加载&#xff08;英语&#xff1a;Extract, transform, load&#xff0c;简称ETL&#xff09;&#xff0c;用来描述将资料从来源端经过抽取、转置、加载至目的端的过程。ETL一词较常用在数据仓库&#xff0c;但其对象并不限于数据仓库。 ETL&…

某知名国企面试题

引言 金九银十&#xff0c;求职热潮再度来袭。最近&#xff0c;有位同学去一家知名国企应聘&#xff0c;回来后带回了一套面试题。这套面试题非常典型&#xff0c;其中包含了许多供应链金融方面的典型问题。这些问题很有分享的价值&#xff0c;大家也可以先自己独立思考一下&a…

PFC和LLC的本质和为什么要用PFC和LLC电路原因

我们可以用电感和电容的特性,以及电压和电流之间的不同步原理来解释PFC(功率因数校正)和LLC(谐振变换器)。 电感和电容的基本概念 电感(Inductor): 电感是一种储存电能的组件。它的电流变化比较慢,电流在电感中延迟,而电压变化得比较快。可以把电感想象成一个“滞后…

接口自动化测试介入项目管理流程

&#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 下图为接口自动化测试介入梧桐项目管理流程图 前景和目标&#xff1a; 现在公司的项目流程都是全部开发完成后提交到测试环境进行测试&#xff0c;导致测试人员在…

基于FPGA的以太网设计(三)

通过前文介绍了RGMII接口时序我们可以知道&#xff0c;RGMII接口是在时钟信号的上升沿和下降沿均进行数据的传输&#xff0c;而FPGA则在时钟的单沿传输数据&#xff0c;因此我们需要编写代码将RGMII接口转换为GMII接口。 由于前面的介绍我们知道RTL8211默认工作在延时状态&…

深入计算机语言之C++:类与对象(上)

&#x1f511;&#x1f511;博客主页&#xff1a;阿客不是客 &#x1f353;&#x1f353;系列专栏&#xff1a;从C语言到C语言的渐深学习 欢迎来到泊舟小课堂 &#x1f618;博客制作不易欢迎各位&#x1f44d;点赞⭐收藏➕关注 前面我们学习了关于c语言的一些基础知识&#xff…

Lucene 倒排索引

倒排索引是什么&#xff1f; 【定义】倒排索引&#xff08;Inverted Index&#xff09;是一种用于信息检索的数据结构&#xff0c;尤其适用于文本搜索。它与传统索引的主要区别在于&#xff0c;传统索引是根据文档来查找词语的位置&#xff0c;而倒排索引则是根据词语来查找文…

穷举vs暴搜vs深搜vs回溯vs剪枝(一)

文章目录 全排列子集找出所有子集的异或总和再求和全排列 II电话号码的字母组合 全排列 题目&#xff1a;全排列 思路 通过深度优先搜索的方式&#xff0c;不断枚举每个数在当前位置的可能性&#xff0c;然后回溯到上一个状态&#xff0c;直到枚举完所有可能性得到正确的结果 r…

双11购物节,淘宝、京东薅羊毛~红包攻略

紧急通知&#xff1a;今年的双11购物节相比去年又提前了&#xff01;为了迎接这一购物盛宴&#xff0c;各大电商平台纷纷推出了红包活动&#xff0c;其中京东和淘宝的红包活动尤为引人注目。以下是小编为各位消费者精心整理的红包攻略。 淘宝双11超级红包 天猫双11超级红包&a…

无人直播自动化回复客户咨询

我们插件是根据页面元素变动进行自动化操作的&#xff0c;想要实现网页版自动化&#xff0c;必须了解html以及dom结构&#xff0c;还有xpath定位方法。 各大直播后台页面结构不一样&#xff0c;所以要进行兼容处理&#xff0c;我们一个插件支持以下直播或客服平台 唯一客服浏…

【机器学习】特征降维|低方差过滤|主成分分析PCA|相关系数法|皮尔逊相关系数|斯皮尔曼相关系数

特征降维 特征降维 为什么要进行特征降维? 特征对训练模型非常重要,当用于训练的数据集包涵一些不重要的特征时,可能会导致模型泛化性能不加 eg&#xff1a;某些特征的取值较为接近&#xff0c;其包含的信息较少eg&#xff1a;希望特征独立存在对预测产生影响&#xff0c;两…

wsl环境下安装Ubuntu,并下载MySQL5.7

安装操作需root权限&#xff0c;切换root用户有两种方式&#xff1a; 1-通过 sudo su - &#xff0c;切换到root用户&#xff08;登录后长期有效&#xff09;。 2-在每一个命令前加上sudo&#xff0c;临时提升权限&#xff08;仅对一条命令有效&#xff09;。 1、下载apt仓库…

轮椅拐杖残疾人检测数据集 4400张 轮椅拐杖 标voc yolo

轮椅拐杖残疾人检测数据集 4400张 轮椅拐杖 标voc yolo 2 分类名: (图片张数&#xff0c; 标注个数) whee Ichair: (3766&#xff0c; 4460) person_ crutch: (682&#xff0c; 693) 总数: (4448&#xff0c; 5153) . 总类(nc): 2类 轮椅拐杖残疾人检测数据集介绍 数据集概述…

Laravel Filament 如何配置多语言支持

演示 一、安装拓展包outerweb/filament-translatable-fields composer require outerweb/filament-translatable-fields配置模型 该套件包含一个名为 HasTranslations 的特性&#xff0c;用于使 Eloquent 模型具备多语言功能。翻译值以 JSON 格式存储&#xff0c;并不需要额外…

【数据采集工具】Flume从入门到面试学习总结

国科大学习生活&#xff08;期末复习资料、课程大作业解析、大厂实习经验心得等&#xff09;: 文章专栏&#xff08;点击跳转&#xff09; 大数据开发学习文档&#xff08;分布式文件系统的实现&#xff0c;大数据生态圈学习文档等&#xff09;: 文章专栏&#xff08;点击跳转&…

ROS2 Jazzy(二) ROS相关工具 概念

以下demo都是出自官方教程urdf_tutorial Link CLI ros2命令行,可以使用ros2 --help来查看指南 ros2 --help # 包括ros2 pkg/topic等等&#xff0c;基础且常用VScode vscode老朋友了&#xff0c;但是要配置好适合ros2开发的vscode&#xff0c;还是有点麻烦的。 配置C语言相关…

ChatTTS在Windows电脑的本地部署与远程生成音频详细实战指南

文章目录 前言1. 下载运行ChatTTS模型2. 安装Cpolar工具3. 实现公网访问4. 配置ChatTTS固定公网地址 前言 本篇文章主要介绍如何快速地在Windows系统电脑中本地部署ChatTTS开源文本转语音项目&#xff0c;并且我们还可以结合Cpolar内网穿透工具创建公网地址&#xff0c;随时随…

使用scss生成旋转圆圈

图片 html代码&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Document</title>…

Windows下MYSQL8.0如何恢复root权限

误操作把root权限清掉导致数据库无法登录&#xff08;确实很难受&#xff09;&#xff0c;在网上找了很多方法&#xff0c;发现没有很行之有效的方法&#xff0c;在多方尝试终于找到了适合敏感宝宝体质的方法。 C:\Users\Administrator>mysql -u root -P3307 ERROR 1045 (2…

通信工程学习:什么是USB通用串行总线

USB&#xff1a;通用串行总线 USB&#xff0c;全称Universal Serial Bus&#xff08;通用串行总线&#xff09;&#xff0c;是一种外部总线标准&#xff0c;用于规范电脑与外部设备的连接和通讯。以下是关于USB的详细介绍&#xff1a; 一、USB的定义与特点 USB的定义&#xff…