IoC+AOP的简单实现

对EnterLib有所了解的人应该知道,其中有一个名叫Policy
Injection的AOP框架;而整个EnterLib完全建立在另一个叫作Unity的底层框架之上,我们可以将Unity看成是一个IoC的框架。对于一个企业应用来说说,AOP和IoC是我们进行逻辑分离和降低耦合度最主要的方式,而将两者结合起来具有重要的现实意义。

一、基于IoC+AOP的编程

到底将IoC和AOP进行整合后,会对编程但来怎样的影响,我写了一个简单的例子(你可以从这里下载该实例)。假设我现在有两个模块,分别称为Foo和Bar,通过如下同名的类来表示。Foo和Bar具有各自的接口,分别为IFoo和IBar。简单起见,我在两个接口中定义了相同的方法:DoSomething。在Foo中,具有一个类型为IBar的只读属性,而DoSomething的实现就是通过调用该属性的同名方法实现。

   1: public interface IFoo
   2: {
   3:     void DoSomething();
   4: }
   5:    
   6: public interface IBar
   7: {
   8:     void DoSomething();
   9: }    
  10: [FooCallHandler]
  11: public class Foo : IFoo
  12: {
  13:     public IBar Bar { get; private set; }
  14:     public Foo(IBar bar)
  15:     { this.Bar = bar; }
  16:     public void DoSomething()
  17:     {
  18:         this.Bar.DoSomething();
  19:     }
  20: }
  21: [BarCallHandler]
  22: public class Bar : IBar
  23: {
  24:     public void DoSomething()
  25:     {
  26:         Console.WriteLine("Do something...");
  27:     }
  28: }

在类型Foo和Bar上分别应用了两个自定义特性:FooCallHandlerAttributeBarCallHandlerAttribute,熟悉PIAB的读者对此应该不感到陌生——CallHandler承载着被分离出来的横切关注点(Crosscutting

Concern)的实现。由于在这里仅仅是一个简单的演示,我定义了两个最简单的CallHandler:FooCallHandler和BarCallHandler。下面是这个两个CallHandler的对应的HandlerAttribute的定义。

   1: public class FooCallHandler : ICallHandler
   2: {
   3:     public IMethodReturn Invoke(IMethodInvocation input, GetNextHandlerDelegate getNext)
   4:     {
   5:         Console.WriteLine("Foo: Preoperation is executed.");
   6:         var methodReturn = getNext()(input, getNext);
   7:         Console.WriteLine("Foo: Postoperation is executed.");
   8:         return methodReturn;
   9:     }
  10:     public int Order { get; set; }
  11: }
  12: public class BarCallHandler : ICallHandler
  13: {
  14:     public IMethodReturn Invoke(IMethodInvocation input, GetNextHandlerDelegate getNext)
  15:     {
  16:         Console.WriteLine("Bar: Preoperation is executed.");
  17:         var methodReturn = getNext()(input, getNext);
  18:         Console.WriteLine("Bar: Postoperation is executed.");
  19:         return methodReturn;
  20:     }
  21:     public int Order { get; set; }
  22: }
  23: public class FooCallHandlerAttribute : HandlerAttribute
  24: {
  25:     public override ICallHandler CreateHandler(IUnityContainer container)
  26:     {
  27:         return new FooCallHandler { Order = this.Order };
  28:     }
  29: }
  30: public class BarCallHandlerAttribute : HandlerAttribute
  31: {
  32:     public override ICallHandler CreateHandler(IUnityContainer container)
  33:     {
  34:         return new BarCallHandler { Order = this.Order };
  35:     }
  36: }

两个CallHandler具有相似的实现:在目标方法执行前后打印相应的文字,代表在对方法调用进行拦截后,需要在方法执行前后实现的横切关注点。目前为止,提供具体功能的组建定义完成,我们来编写消费它们的客户程序:

   1: static void Main(string[] args)
   2: {
   3:     var container = GetContainer();           
   4:     var foo = container.Resolve<IFoo>();
   5:     foo.DoSomething();
   6: }

从上面我们可以看出,我们最终的代码仅仅只要三行。虽然简单,我们不妨来也做一下分析:首先,客户端对组件Foo的调用是基于接口(IFoo)而不是基于具体类型Foo,这样能够尽可能地降低对组件Foo的依赖;其次,Foo依赖于Bar,而这种依赖也是基于接口的,这在Foo的定义了看得出来;第三,虽然Foo依赖于Bar,但是这种依赖是和Foo的具体实现相关,和客户程序无关,所以客户程序未曾出现Bar和IBar的影子。

运行上面的程序,你会得到如下的输出。从中我们可以看出:不但具体的业务功能(即定义在Bar的DoSomething方法中的逻辑)能够正常执行,通过自定义特性的方式应用到两个组件上动态注入的横切关注点也得到正确地执行

   1: Foo: Preoperation is executed.
   2: Bar: Preoperation is executed.
   3: Do something...
   4: Bar: Postoperation is executed.
   5: Foo: Postoperation is executed.

那么,是什么导致了程序的完全按照我们希望的方式执行的呢?由于客户端逻辑只有三句代码,你肯定会猜的到:所有的一切都是由一开始创建的container对象完成的。如何你了解Unity的话,应该可以猜出这是一个UnityContainer。通过接口和类型的匹配关系的注册,UnityContainer知道如何根据接口找到相应的实现类型(IFoo-〉Foo,IBar-〉Bar),这不难理解,这也不是本篇文章介绍的重点。我们关注点是:UnityContainer是如何让通过自定义特性方式应用在Foo和Bar上的两个CallHandler得到执行的?

二、CallHandler是如何被执行的

同所有的AOP实现一样,PIAB也是采用方法拦截(Method Interception)机制。具体来说,PIAB又具有两个不同的方式:实例拦截(Instance Interception)和类型拦截(Type Interception)。而前者具有两种不同的实现:TransparentProxy/RealProxy的方式Reflection Emit的方式。三种不同拦截方式的具体实现,不是本文的重点,对此有兴趣的朋友可以参阅PIAB官方文档。

总的来说,能够在方法执行过程中被拦截的对象需要通过PIAB的方式进行创建,实际上PIAB是对目标对象进行相应的封装,进而得到一个代理对象。由于方法调用请求是从代理对象发出,PIAB在将该请求发送给最终的目标对象之前将其拦截下来,从而使相应的CallHandler得以执行。那么,如果我们能够让UnityContainer在对象创建过程完成这一道工序,那么最终被创建出来的对象就具有了被拦截的能力

如何将PIAB对实例的封装操作注入到UnityContainer怎个对象创建流程中呢?这需要借助于UnityContainer提供的扩展机制。虽然Unity仅仅是一个轻量级的IoC框架,但是内部的实现其实是挺复杂的。个人曾经不止一次地分析过Unity的源代码,但是没过多久就忘得七七八八。不过,万变不离其宗,UnityContainer最根本的就是其BuilderStrategy管道(可以参阅我的文章《你知道Unity IoC Container是如何创建对象的吗?》)。

我们的解决方案就是将PIAB对实例的封装写在相应的BuilderStrategy种,然后通过UnityContainerExtension注册到某个UnityContainer中。为此我创建了如下一个名称为InterceptionStrategy的BuilderStrategy。

   1: public class InterceptionStrategy : BuilderStrategy
   2: {
   3:     public static IUnityContainer UnityContainer { get; private set; }
   4:     public static InjectionMember InterceptionMember { get; private set; }
   5:  
   6:     static InterceptionStrategy()
   7:     {
   8:         IUnityContainer container = new UnityContainer();
   9:         UnityContainerConfigurator configurator = new UnityContainerConfigurator(container);
  10:         EnterpriseLibraryContainer.ConfigureContainer(configurator, ConfigurationSourceFactory.Create());
  11:         UnityContainer = container;
  12:         InterceptionMember = new InstanceInterceptionPolicySettingInjectionMember(new TransparentProxyInterceptor());
  13:     }
  14:  
  15:     public override void PostBuildUp(IBuilderContext context)
  16:     {
  17:         if (null == context.Existing ||
  18:             context.Existing.GetType().FullName.StartsWith("Microsoft.Practices") ||
  19:             context.Existing is IInterceptingProxy)
  20:         {
  21:             return;
  22:         }
  23:         context.Existing = UnityContainer.Configure<TransientPolicyBuildUpExtension>().BuildUp
  24:             (context.OriginalBuildKey.Type, context.Existing, null, InterceptionMember);
  25:     }
  26: }

如果你对Unity的内部机制有一定了解,理解上面的实现应该不成问题,但是你对此一无所知,我讲解得再详细你可能也弄不清楚。所以,对于上述的逻辑实现,在这里就不多作介绍了。

自定义的BuilderStrategy一般需要通过相应的UnityContainerExtension完成注册。下面是我们创建的UnityContainerExtension:InterceptionExtension。我们将UnityContainerExtension对象添加到UnityContainer的BuilderStrategy管道之中,相应的BuilderStage设置为PreCreation。

   1: public class InterceptionExtension: UnityContainerExtension
   2: {
   3:     protected override void Initialize()
   4:     {
   5:         Context.Strategies.AddNew<InterceptionStrategy>(UnityBuildStage.PreCreation);
   6:     }
   7: }

三、GetContainer()的实现…

最初例子得以正常运行的魔力来自于通过静态方法GetContainer创建的UnityContainer对象,我们现在来看看该方法的实现。首先我们创建一个UnityContainer对象,然后对其进行初始化配置,最后将上面创建的InterceptionExtension扩展添加到该UnityContainer中。接口和实现类型的注册被随后执行,不过在真正的开发中,我们习惯通过配置文件进行注册。这就是整个实现,没有复杂的逻辑,却能带来很大的用处。

   1: static IUnityContainer GetContainer()
   2: {
   3:     IUnityContainer container = new UnityContainer();
   4:     UnityContainerConfigurator configurator = new UnityContainerConfigurator(container);
   5:     EnterpriseLibraryContainer.ConfigureContainer(configurator, ConfigurationSourceFactory.Create());
   6:     container.AddNewExtension<InterceptionExtension>();
   7:     container.RegisterType<IFoo, Foo>();
   8:     container.RegisterType<IBar, Bar>();
   9:     return container;
  10: }

四、申明

文中提供解决方案只是为你提供一种思路,相关的逻辑实现基本来自于脑子中的灵光一现,并没有进行深入的评估和性能测试。如果你希望在你自己的项目中使用,最好在此基础上进行深入的思考,相信会发现其中存在的不足。此外,不知道读者有没有注意,上面的实现方式仅仅提供一种拦截方式,即基于TransparentProxy的方式,有兴趣的读者可以再作一些扩展,实现对拦截方式的定制。

本文所介绍的实现实际上是采用EnterLib PIAB的机制。由于PIAB本身的问题,上面的实现只能支持构造器注入,对属性注入和方法注入不提供支持,详情请参考《EnterLib PIAB又一个BUG?[续]——这是一个致命的BUG

作者:蒋金楠
微信公众账号:大内老A
微博:www.weibo.com/artech
如果你想及时得到个人撰写文章以及著作的消息推送,或者想看看个人推荐的技术资料,可以扫描左边二维码(或者长按识别二维码)关注个人公众号(原来公众帐号蒋金楠的自媒体将会停用)。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

原文链接

时间: 2024-10-03 12:26:29

IoC+AOP的简单实现的相关文章

一起谈.NET技术,IoC+AOP的简单实现

对EnterLib有所了解的人应该知道,其中有一个名叫Policy Injection的AOP框架:而整个EnterLib完全建立在另一个叫作Unity的底层框架之上,我们可以将Unity看成是一个IoC的框架.对于一个企业应用来说说,AOP和IoC是我们进行逻辑分离和降低耦合度最主要的方式,而将两者结合起来具有重要的现实意义. 一.基于IoC+AOP的编程 到底将IoC和AOP进行整合后,会对编程但来怎样的影响,我写了一个简单的例子(你可以从这里下载该实例).假设我现在有两个模块,分别称为Fo

第二章 IoC 概念与简单的使用

2.1 IoC的概念         Inverseof Control,控制反转是Spring容器的内核,AOP.声明式事务等功能都是在此基础上进行的.         IoC主要功能是依赖关系的转移.应用的本身不负责依赖对象的创建和维护,而是由Spring容器负责.控制权就由应用转移到了外部容器.         IoC的主要功能由控制反转来解释并不是很好理解.所以提出了新的概念DependencyInjection.         DI依赖注入,调用类对某一接口实现类的依赖关系由第三方(

AOP与IOC的冲突/aop无法注入

问题描述 AOP与IOC的冲突/aop无法注入 项目架构同时搭建spring 和 spring mvc在分库架构体系下我打算用aop来实现分库的实现流程,我在spring xml里扫描了service,在mvc xml里扫描了controller于是在mvc里配置aop 切面aspect可以设定service为切入目标但是在spring里配置aop 切面aspect既不能设定service也不能设置controller为切入点注:使用了@controller @service注解 问题在于,无论

Spring的IoC容器实现原理(一)#loadBeanDefinition

Spring有十几个组件,核心组件为bean(演员)-context(舞台)-core(道具) bean包装的是object,而object中肯定要有数据,如何给这些数据提供生存环境就是context要解决的问题,对于context来说他就是要发现每个bean之间的关系,为他们建立起来并维护好这些关系.所以context就是一个bean关系的集合,这个关系集合就是我们常说的IOC容器.core组件就是发现.建立和维护每个bean之间的关系所需要的一些工具,把core叫做util更为贴切.   

AOP编程入门--Java篇

编程 Aspect Oriented Programming(AOP),面向切面编程,是一个比较热门的话题.AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效果.比如我们最常见的就是日志记录了,举个例子,我们现在提供一个服务查询学生信息的,但是我们希望记录有谁进行了这个查询.如果按照传统的OOP的实现的话,那我们实现了一个查询学生信息的服务接口(StudentInfoService)和其实现类(Student

从spring的IOC说起(一)

这段时间也着实好好的看了下spring的相关书籍,对其也有了大概和初步的 认识和理解,虽然之前也一直听说spring是一个非常优秀的开源框架,可一直没 有机会学习和使用(是不是有点落伍了?呵呵),所以呢,这段时间就重点学习 了spring(一个星期的时间当然是入门级的啦~~) 大家一直都说spring的IOC如何如何的强大,其实我倒觉得不是IOC如何的强 大,说白了IOC其实也非常的简单.我们先从IOC说起,这个概念其实是从我们平 常new一个对象的对立面来说的,我们平常使用对象的时候,一般都是

Spring.net(二)----初探IOC容器

我在上一篇关于Spring.net的文章"Spring.NET框架简介及模块说明 "中很详细的介绍了,本文就不旧话从提.我门就直奔主题吧. 1.首先了解两个接口. IObjectFactory接口和IApplicationContext接口:他两个称为"容器"或"IOC容器". Spring.net框架的核心原则是非侵入性. IObjectFactory接口是初始化.配置及管理对象的实际容器. IObjectFactory全限定名为Spring.

IoC在ASP.NET Web API中的应用

控制反转(Inversion of Control,IoC),简单地说,就是应用本身不负责依赖对象的创建和维护,而交给一个外部容器来负责.这样控制权就由应用转移到了外部IoC容器,控制权就实现了所谓的反转.比如在类型A中需要使用类型B的实例,而B实例的创建并不由A来负责,而是通过外部容器来创建.通过IoC的方式实现针对目标HttpController的激活具有重要的意义.[本文已经同步到<How ASP.NET Web API Works?>] 一. 基于IoC的HttpControllerA

利用Spring IOC技术实现用户登录验证机制_java

利用 Spring IOC 技术实现用户登录的验证机制,对用户进行登录验证. 首先利用 Spring 的自动装配模式将 User 对象注入到控制器中,然后将用户输入的用户名和密码与系统中限定的合法用户的用户名和密码进行匹配. 当用户名与密码匹配成功时,跳转到登录成功页面:当用户名与密码不匹配时,跳转到登录失败的页面. 1.创建 User 对象,定义用户名和密码属性,代码如下: package com.importnew; public class User { private String us