本文简单告诉大家如何在 .NET Core 里面使用 AOP 切面编程,使用 DispatchProxy 做任意接口的动态代理实现
使用 DispatchProxy 有一些限制,如只能创建接口的实例。使用 DispatchProxy 可以让咱做到从无中创建出某个实例,这个实例继承咱想要的接口
首先需要创建某个动态代理类继承 DispatchProxy 类,例如创建 Foo 类。此时需要实现 Invoke 接口,这个接口的含义是当代码调用接口里面的函数,包括属性的设置和获取函数的时候将会进入 Invoke 方法
protected override object Invoke(MethodInfo targetMethod, object[] args)
{
// 忽略代码
}
因此通过 Invoke 方法可以假装自己是对应的接口的实现,可以进行随意更改执行逻辑以及修改返回值
而 DispatchProxy 另一个核心方法是静态的 Create 方法,这个方法要求传入两个泛形,第一个类型是想要创建出来的实例继承的接口,第二个类型是某个继承 DispatchProxy 的类
例如咱定义了一个 IF1 的接口,此时可以在 Foo 添加如下代码创建出一个不存在的类型实例,这个实例继承了 IF1 接口
public class Foo : DispatchProxy
{
public static T GetObject<T>()
{
return DispatchProxy.Create<T, Foo>();
}
protected override object Invoke(MethodInfo targetMethod, object[] args)
{
return "lindexi";
}
}
假设 IF1 的定义如下
interface IF1
{
string F2();
}
此时的使用方法如下
Console.WriteLine(Foo.GetObject<IF1>().F2());
执行代码可以看到输出了 lindexi
也就是实际调用了 Foo 方法
可以看到 DispatchProxy 的作用就是提供静态方法用于创建继承指定接口的对象,同时让继承 DispatchProxy 的类可以拿到接口所调用的方法,用来修改执行逻辑和返回值
一个实际的使用的例子,提供了调用某个对象的每个方法之前给这个对象调用方法记日志,这是最简便的记日志的方法了,尽管性能很渣
public class LoggingAdvice<T> : DispatchProxy
{
private T Object { set; get; }
public static T CreateLogging(Func<T> creator)
{
object proxy = DispatchProxy.Create<T, LoggingAdvice<T>>();
((LoggingAdvice<T>)proxy).Object = creator();
return (T)proxy;
}
protected override object Invoke(MethodInfo targetMethod, object[] args)
{
Console.WriteLine($"开始执行 {targetMethod.Name}");
var result = targetMethod.Invoke(Object, args);
Console.WriteLine($"执行完成 {targetMethod.Name}");
return result;
}
}
此时就能做到在方法执行前后添加日志,如这里有一个接口和一个类,使用方法请参考以下示例
class Program
{
static void Main(string[] args)
{
var foo = LoggingAdvice<IF1>.CreateLogging(() => new Foo());
foo.F2();
}
}
interface IF1
{
string F2();
}
class Foo : IF1
{
/// <inheritdoc />
public string F2()
{
return "lindexi";
}
}
参考
本文会经常更新,请阅读原文: https://dotnet-campus.github.io//post/dotnet-core-2-%E4%BD%BF%E7%94%A8-DispatchProxy-%E5%81%9A%E5%8A%A8%E6%80%81%E4%BB%A3%E7%90%86-AOP-%E5%85%A5%E9%97%A8.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 lindexi (包含链接: https://dotnet-campus.github.io/ ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 。