.NET静态代码织入——肉夹馍(Rougamo) 发布1.2.0
肉夹馍(https://github.com/inversionhourglass/Rougamo)通过静态代码织入方式实现AOP的组件,其主要特点是在编译时完成AOP代码织入,相比动态代理可以减少应用启动的初始化时间让服务更快可用,同时还能对静态方法进行AOP。
在 1.0.0 版本中,肉夹馍提供了最基础的AOP功能,可以进行日志记录和APM埋点。在 1.1.0 版本中新增了对更加实用的AOP操作的支持,可以进行异常处理和修改返回值。本次的 1.2.0 版本没有新增功能,主要是对 1.1.0 版本的增强,新增了一个ExMoAttribute
,这个Attribute可能会替代MoAttribute
成为大家更常用的Attribute.
前言
这次无法直接从快速开始入手了,了解一下前因后果会让你对肉夹馍最初的设定和ExMoAttribute
的出现所解决的问题有更清晰的认识,这样也方便后续使用时能够明确的知道是应该使用MoAttribute
还是ExMoAttribute
.
在1.1.0版本发布后陆续收到两个issue,都是反馈在没有使用async语法的Task/ValueTask
返回值的方法无法正确的在方法执行成功(OnSuccess)和方法退出前(OnExit)执行织入的代码,比如记录方法的执行耗时的示例:
static async Task Main(string[] args)
{
await Test();
}
[Timeline]
static Task Test()
{
Console.WriteLine($"{nameof(Test)} start");
return Task.Run(() =>
{
Thread.Sleep(1000);
Console.WriteLine($"{nameof(Test)} end");
});
}
class TimelineAttribute : MoAttribute
{
private Stopwatch _stopwatch;
public override void OnEntry(MethodContext context)
{
_stopwatch = Stopwatch.StartNew();
Console.WriteLine($"{context.Method.Name} {nameof(OnEntry)}");
}
public override void OnExit(MethodContext context)
{
_stopwatch.Stop();
Console.WriteLine($"{context.Method.Name} {nameof(OnExit)} - {_stopwatch.ElapsedMilliseconds}ms");
}
}
期望的输出可能是下面这样,等Test
方法返回值Task
执行完成之后再执行OnExit
,统计耗时到返回的Task
执行完毕之后:
Test OnEntry
Test start
Test end
Test OnExit - 1096ms
而实际的输出是下面这样的,方法的耗时仅为Task
对象创建后返回的执行耗时,并没有等待Task
执行:
Test OnEntry
Test start
Test OnExit - 96ms
Test end
这种表现其实是最开始设计时的设定。在我们刚接触async/await语法时,我们或许有听到这样的介绍“async/await让我们像写同步方法那样去写异步方法”。是的,有了async/await,我们就不用像以前EAP/APM那样去编写callback了,代码整体看起来和同步代码无异,同时我们也会注意到一点,异步方法的返回值类型是Task/ValueTask
,但是我们实际return的对象类型却是其泛型参数类型(Task/ValueTask中的那个T),肉夹馍采用了这一设定。所以对于返回值时Task/ValueTask
的方法,如果使用了async/await语法,那么你通过MethodContext.RealReturnType
获取到的返回值类型就是其泛型参数类型(没有泛型参数时就是void
),同时通过MethodContext.HandledException
和MethodContext.ReplaceReturnValue
设置/修改返回值时,返回值的类型也是Task/ValueTask
的泛型参数类型。而对于没有使用async/await语法的方法,那么返回值类型就是Task/ValueTask
本身。也可以简单的理解为MoAttribute
里采用的方法返回值类型与你编写代码时return的对象类型相同。也因为这样的设定,在上面的示例中由于没有使用async/await语法,其实际的返回值类型就是Task
,并不会去等待Task
执行完毕,所以有了上面那段代码的执行效果。
如果希望上面那段代码达到预期的效果,有没有什么方案呢?答案是:有的
// 仅对TimelineAttribute的OnExit方法进行改造
class TimelineAttribute : MoAttribute
{
// ...
public override void OnExit(MethodContext context)
{
if (typeof(Task).IsAssignableFrom(context.RealReturnType))
{
((Task)context.ReturnValue).ContinueWith(t => _OnExit());
}
else
{
_OnExit();
}
void _OnExit()
{
_stopwatch.Stop();
Console.WriteLine($"{context.Method.Name} {nameof(OnExit)} - {_stopwatch.ElapsedMilliseconds}ms");
}
}
}
上面的方案通过自行判断返回值类型,对继承自Task
的返回值通过显式转换后调用ContinueWith
达到需要的效果。这个思路是通用的,但对有些需求实现起来就比较麻烦并且需要对肉夹馍的执行逻辑有一定的了解,比如异常处理,上面的例子中如果在Task.Run
之前抛出异常,你需要在OnException
中进行异常处理,而如果是在Task.Run
里的Action
中抛出异常,那就需要到OnSuccess
中通过ContinueWith
判断Task
是否执行异常:
[Timeline]
static Task Test()
{
Console.WriteLine($"{nameof(Test)} start");
// throw new Exception(); // 这里抛出异常在OnException中处理
return Task.Run(() =>
{
// throw new Exception(); // 这里抛出异常在OnSuccess中通过ContinueWith处理
Thread.Sleep(1000);
Console.WriteLine($"{nameof(Test)} end");
});
}
在两个issue提出之后,细细想来这种需求或许才是大家最常用的,有时不使用async语法,可能仅仅是因为方法的重载只需对参数稍作处理然后直接调用重载方法即可,这种情况下不使用async语法也是很正常的。针对这类情况,就有了本次版本推出的ExMoAttribute
了。
ExMoAttribute
ExMoAttribute
的目标是解决前面提到的问题,对没有使用async语法的方法采用MoAttribute
中使用了async语法相同的逻辑。
快速开始
# 添加NuGet引用
dotnet add package Rougamo.Fody
class TimelineAttribute : ExMoAttribute
{
private Stopwatch _stopwatch;
protected override void ExOnEntry(MethodContext context)
{
_stopwatch = Stopwatch.StartNew();
Console.WriteLine($"{context.Method.Name} {nameof(OnEntry)}");
}
protected override void ExOnExit(MethodContext context)
{
_stopwatch.Stop();
Console.WriteLine($"{context.Method.Name} {nameof(OnExit)} - {_stopwatch.ElapsedMilliseconds}ms");
}
}
TimelineAttribute
改成上面的代码即可完成最初统计耗时的需求了,应用了该Attribute的方法无论是否使用async语法都能达到同样的效果,再也不用去判断MethodContext.RealReturnType
了。
ExMoAttribute的使用差异(重要)
ExMoAttribute
和MoAttribute
除了类名和方法名有所区别之外,在使用时也有些许区别,下面是使用ExMoAttribute
时与之前不同的地方:
- 使用
MethodContext.ExReturnValue
获取方法返回值。如果是没有使用async语法的方法,使用之前的MethodContext.ReturnValue
获取返回值,你获取到的会是Task/ValueTask
类型的返回值,而不是其泛型参数类型; - 使用
MethodContext.ExReturnType
获取返回值类型。如果你要修改返回值或处理异常,你设置的返回值类型需要与MethodContext.ExReturnType
相同; - 使用
MethodContext.ExReturnValueReplaced
获取返回值是否被修改。如果是没有使用async语法的方法,使用之前的MethodContext.ReturnValueReplaced
获取到的一直都会是true
,因为返回值被ContinueWith
返回的Task替换了。
当前版本除了上面三个属性在使用时需要注意,其他的与MoAttribute
基本无异,包括处理异常时依旧调用MethodContext.HandledException
方法,修改/设置返回值时依旧调用MethodContext.ReplaceReturnValue
方法,后续还会不会有其他差异就需要大家关注一下版本日志了。
.NET静态代码织入——肉夹馍(Rougamo) 发布1.2.0的更多相关文章
- .NET静态代码织入——肉夹馍(Rougamo) 发布1.1.0
肉夹馍(https://github.com/inversionhourglass/Rougamo)通过静态代码织入方式实现AOP的组件,其主要特点是在编译时完成AOP代码织入,相比动态代理可以减少应 ...
- .NET静态代码织入——肉夹馍(Rougamo)
肉夹馍是什么 肉夹馍通过静态代码织入方式实现AOP的组件..NET常用的AOP有Castle DynamicProxy.AspectCore等,以上两种AOP组件都是通过运行时生成一个代理类执行AOP ...
- 30个类手写Spring核心原理之AOP代码织入(5)
本文节选自<Spring 5核心原理> 前面我们已经完成了Spring IoC.DI.MVC三大核心模块的功能,并保证了功能可用.接下来要完成Spring的另一个核心模块-AOP,这也是最 ...
- Spring的LoadTimeWeaver(代码织入)
在Java 语言中,从织入切面的方式上来看,存在三种织入方式:编译期织入.类加载期织入和运行期织入.编译期织入是指在Java编译期,采用特殊的编译器,将切面织入到Java类中:而类加载期织入则指通过特 ...
- Spring的LoadTimeWeaver(代码织入)(转)
https://www.cnblogs.com/wade-luffy/p/6073702.html 在Java 语言中,从织入切面的方式上来看,存在三种织入方式:编译期织入.类加载期织入和运行期织入. ...
- 【开源】.Net Aop(静态织入)框架 BSF.Aop
BSF.Aop .Net 免费开源,静态Aop织入(直接修改IL中间语言)框架,类似PostSharp(收费): 实现前后Aop切面和INotifyPropertyChanged注入方式. 开源地址: ...
- Java AOP (1) compile time weaving 【Java 切面编程 (1) 编译期织入】
According to wikipedia aspect-oriented programming (AOP) is a programming paradigm that aims to inc ...
- AOP静态代理解析2-代码织入
当我们完成了所有的AspectJ的准备工作后便可以进行织入分析了,首先还是从LoadTimeWeaverAwareProcessor开始. LoadTimeWeaverAwareProcessor实现 ...
- 框架源码系列三:手写Spring AOP(AOP分析、AOP概念学习、切面实现、织入实现)
一.AOP分析 问题1:AOP是什么? Aspect Oriented Programming 面向切面编程,在不改变类的代码的情况下,对类方法进行功能增强. 问题2:我们需要做什么? 在我们的框架中 ...
随机推荐
- JetBrains系列软件激活
1.将以下记录加入hosts文件 0.0.0.0 account.jetbrains.com0.0.0.0 www.jetbrains.com 2.激活方式选择licence server,填写以下激 ...
- 分享|2022数字安全产业大数据白皮书(附PDF)
内容摘要: 2021年以来,数字安全赛道的受关注程度达到一个历史新高度.<数据安全法><个人信息保护法><关键信息基础设施安全保护条例>,一个接一个重磅的法规接连出 ...
- PE格式: 分析IatHook并实现
Ring 3层的 IAT HOOK 和 EAT HOOK 其原理是通过替换IAT表中函数的原始地址从而实现Hook的,与普通的 InlineHook 不太一样 IAT Hook 需要充分理解PE文件的 ...
- k8s QoS与pod驱逐
概述 QoS是Quality of Service的缩写,即服务质量.每个pod属于某一个QoS分类,而Kubernetes会根据pod的QoS级别来决定pod的调度.抢占调度和驱逐优先级,而且pod ...
- 记一次Linux server偶发CPU飙升问题的跟进与解决
背景 进入6月后,随着一个主要功能版本api的上线,服务端的QPS翻了一倍,平时服务器的CPU使用稳定在30%上下,高峰期则在60%上下,但是偶尔会有单台机器出现持续数分钟突然飙到90%以上,导致大量 ...
- 天人合一物我相融,站点升级渐进式Web应用PWA(Progressive Web Apps)实践
原文转载自「刘悦的技术博客」https://v3u.cn/a_id_216 PWA(Progressive web apps,渐进式 Web 应用)使用现代的 Web API 以及传统的渐进式增强策略 ...
- Windows 电脑杀毒简单有效的方式
Windows 电脑杀毒通常会选择杀毒软件,这样太笨重,且容易占内存和存在流氓软件侵入. 推荐使用 Windows 自带的恶意软件删除工具 按住 Win + R 键,弹出运行窗口,输入 mrt. 系统 ...
- ceph 006 rbd高级特性 rbd快照 镜像克隆 rbd缓存 rbd增量备份 rbd镜像单向同步
版本 [root@clienta ~]# ceph -v ceph version 16.2.0-117.el8cp (0e34bb74700060ebfaa22d99b7d2cdc037b28a57 ...
- Hive存储格式之RCFile详解,RCFile的过去现在和未来
我在整理Hive的存储格式和压缩格式,本来打算一篇发出来,结果其中一小节就有很多内容,于是打算写成Hive存储格式和压缩格式系列. 本节主要讲一下Hive存储格式最早的典型的列式存储格式RCFile. ...
- java学习第七天xml.day18
反射 在java中,反射主要是指程序可以访问.检测和修改它本身状态或行为的一种能力. 获取字节码的方式: 使用反射获取构造器 : 内省