.NET 通过 Autofac 和 DynamicProxy 实现AOP

  什么是AOP?引用百度百科:AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。实现AOP主要由两种方式,一种是编译时静态植入,优点是效率高,缺点是缺乏灵活性,.net下postsharp为代表者(这个是收费的)。另一种方式是动态代理,优缺点与前者相反,动态为目标类型创建代理,通过代理调用实现拦截。AOP能做什么,常见的用例是事务处理、日志记录等等。下面就讲讲Autofac怎么实现AOP,Autofac是一个.net下非常优秀,性能非常好的IOC容器(.net下效率最高的容器),加上AOP简直是如虎添翼。Autofac的AOP是通过Castle(也是一个容器)项目的核心部分实现的,名为Autofac.Extras.DynamicProxy,顾名思义,其实现方式为动态代理。

  使用前的准备:

    通过Nuget安装程序包 :Autofac、Autofac.Extras.DynamicProxy,安装成功之后会增加三个引用

    

  下面正式开始了!

 

  第一步:创建拦截器

  下面是一个简单的拦截器示例,该拦截器的功能是显示被拦截的方法名称、参数列表和返回结果

  

 1  /// <summary>
 2     /// 拦截器 需要实现 IInterceptor接口 Intercept方法
 3     /// </summary>
 4     public class CallLogger: IInterceptor
 5     {
 6         TextWriter _output;
 7 
 8         public CallLogger(TextWriter output)
 9         {
10             _output = output;
11         }
12 
13         /// <summary>
14         /// 拦截方法 打印被拦截的方法执行前的名称、参数和方法执行后的 返回结果
15         /// </summary>
16         /// <param name="invocation">包含被拦截方法的信息</param>
17         public void Intercept(IInvocation invocation)
18         {
19             
20             _output.WriteLine("你正在调用方法 \"{0}\"  参数是 {1}... ",
21               invocation.Method.Name,
22               string.Join(", ", invocation.Arguments.Select(a => (a ?? "").ToString()).ToArray()));
23 
24             //在被拦截的方法执行完毕后 继续执行
25             invocation.Proceed();
26 
27             _output.WriteLine("方法执行完毕,返回结果:{0}", invocation.ReturnValue);
28         }
29     }

 

  第二步:注册拦截器到Autofac容器

  拦截器必须注册到Aufofac容器中,可以通过拦截器类型或者命名注入,这两种方式会让使用拦截器的方法有所不同(后面会讲到)。

  

1             // 命名注入
2             builder.Register(c => new CallLogger(Console.Out))
3                    .Named<IInterceptor>("log-calls");
4 
5             // 类型注入
6             builder.Register(c => new CallLogger(Console.Out));    

  第三步:启用拦截器

  启用拦截器主要有两个方法:EnableInterfaceInterceptors(),EnableClassInterceptors()。

  EnableInterfaceInterceptors方法会动态创建一个接口代理

  EnableClassInterceptors方法会创建一个目标类的子类代理类,这里需要注意的是只会拦截虚方法,重写方法

  启用拦截器示例代码:

  

            //启用类代理拦截
            builder.RegisterType<Circle>().EnableClassInterceptors();
            //启用接口代理拦截
            builder.RegisterType<Circle>().EnableInterfaceInterceptors();

  第四步:指明要拦截的类型

  有两种方法:

    第一种:给类型加上特性Attribute

    

    第二种:在注册类型到容器的时候动态注入拦截器

    

1             //动态注入拦截器CallLogger
2             builder.RegisterType<Circle>().InterceptedBy(typeof(CallLogger)).EnableClassInterceptors();

  第五步:测试效果了

    1.类代理拦截

    

    Circle类代码:

    

     2.接口代理拦截

    

     IShape接口代码:

    

1 public interface IShape
2     {
3         /// <summary>
4         /// 形状的面积
5         /// </summary>
6         void Area();
7 
8     }

    Circle类代码:

    

1 public class Circle:IShape
2     {
3         //重写父类抽象方法
4         public void Area()
5         {
6             Console.WriteLine("你正在调用圆求面积的方法");
7         }
8     }

 

    如果有什么地方写得不对欢迎批评改正,如果有什么疑问,欢迎提问。

好的,下面是一个使用 Autofac 读取配置文件并动态注入 IOC 的示例代码(基于 .NET Framework 4.6.1)。 首先,我们需要安装 Autofac 的 NuGet 包。在 Visual Studio 中,可以通过 NuGet 包管理器搜索并安装 Autofac。 然后,我们需要创建一个配置文件来定义我们的依赖项。假设我们的配置文件是一个 JSON 文件,如下所示: ```json { "dependencies": [ { "interface": "MyNamespace.IMyService", "type": "MyNamespace.MyService" } ] } ``` 在这个配置文件中,我们定义了一个名为 `dependencies` 的数组,其中包含一个或多个对象。每个对象都具有两个属性:`interface` 和 `type`。`interface` 属性指定依赖项的接口类型,`type` 属性指定依赖项的实现类型。 接下来,我们需要编写代码来读取配置文件并注册依赖项。下面是一个简单的示例: ```csharp using System.IO; using System.Reflection; using Autofac; using Newtonsoft.Json; public static class ContainerBuilderExtensions { public static void RegisterDependenciesFromConfigFile(this ContainerBuilder builder, string configFile) { var assembly = Assembly.GetExecutingAssembly(); var configPath = Path.Combine(Path.GetDirectoryName(assembly.Location), configFile); var json = File.ReadAllText(configPath); var dependencies = JsonConvert.DeserializeObject<dynamic>(json).dependencies; foreach (var dependency in dependencies) { var interfaceType = Type.GetType(dependency.interface.Value); var implementationType = Type.GetType(dependency.type.Value); builder.RegisterType(implementationType).As(interfaceType); } } } ``` 这个扩展方法将会读取指定的配置文件,并将其中定义的依赖项注册到 Autofac 的容器中。我们可以在应用程序的启动代码中调用这个方法,以便动态注入依赖项。 ```csharp var builder = new ContainerBuilder(); builder.RegisterDependenciesFromConfigFile("dependencies.json"); var container = builder.Build(); // 从容器中解析需要的依赖项 var service = container.Resolve<IMyService>(); ``` 在这个示例中,我们首先创建了一个 `ContainerBuilder` 对象,并注册了一个扩展方法 `RegisterDependenciesFromConfigFile`。然后,我们在应用程序的启动代码中调用这个方法,并使用 `Build` 方法构建一个 Autofac 容器。最后,我们可以通过调用 `Resolve` 方法来从容器中解析需要的依赖项。 注意,我们在这个示例中使用了 Newtonsoft.Json 库来解析 JSON 配置文件。我们还使用了 `Assembly.GetExecutingAssembly().Location` 来获取当前执行程序的路径,然后使用 `Path.Combine` 和 `File.ReadAllText` 方法来读取配置文件。你也可以使用其他方法来读取配置文件,例如使用 `ConfigurationManager` 类或使用 `System.Text.Json` 库。 希望这个示例能够帮助你了解如何使用 Autofac 读取配置文件并动态注入依赖项。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值