dotnet core 2 使用 DispatchProxy 做动态代理 AOP 入门

2020-07-07 15:50:13 浏览数 (1)

本文简单告诉大家如何在 .NET Core 里面使用 AOP 切面编程,使用 DispatchProxy 做任意接口的动态代理实现

使用 DispatchProxy 有一些限制,如只能创建接口的实例。使用 DispatchProxy 可以让咱做到从无中创建出某个实例,这个实例继承咱想要的接口

首先需要创建某个动态代理类继承 DispatchProxy 类,例如创建 Foo 类。此时需要实现 Invoke 接口,这个接口的含义是当代码调用接口里面的函数,包括属性的设置和获取函数的时候将会进入 Invoke 方法

代码语言:javascript复制
        protected override object Invoke(MethodInfo targetMethod, object[] args)
        {
            // 忽略代码
        }

因此通过 Invoke 方法可以假装自己是对应的接口的实现,可以进行随意更改执行逻辑以及修改返回值

而 DispatchProxy 另一个核心方法是静态的 Create 方法,这个方法要求传入两个泛形,第一个类型是想要创建出来的实例继承的接口,第二个类型是某个继承 DispatchProxy 的类

例如咱定义了一个 IF1 的接口,此时可以在 Foo 添加如下代码创建出一个不存在的类型实例,这个实例继承了 IF1 接口

代码语言:javascript复制
    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 的定义如下

代码语言:javascript复制
    interface IF1
    {
        string F2();
    }

此时的使用方法如下

代码语言:javascript复制
            Console.WriteLine(Foo.GetObject<IF1>().F2());

执行代码可以看到输出了 lindexi 也就是实际调用了 Foo 方法

可以看到 DispatchProxy 的作用就是提供静态方法用于创建继承指定接口的对象,同时让继承 DispatchProxy 的类可以拿到接口所调用的方法,用来修改执行逻辑和返回值

一个实际的使用的例子,提供了调用某个对象的每个方法之前给这个对象调用方法记日志,这是最简便的记日志的方法了,尽管性能很渣

代码语言:javascript复制
    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;
        }
    }

此时就能做到在方法执行前后添加日志,如这里有一个接口和一个类,使用方法请参考以下示例

代码语言:javascript复制
    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";
        }
    }

参考

在.NET Core中使用DispatchProxy“实现”非公开的接口 - LamondLu - 博客园

使用.net core中的类DispatchProxy实现AOP - 欧阳.NET - 博客园

0 人点赞