©2007BrillianceTechnologyAllrightsreserved.Chap3创建切面2©2007BrillianceTechnologyAllrightsreserved.议题AOP介绍创建通知定义切入点使用ProxyFactoryBean自动代理3©2007BrillianceTechnologyAllrightsreserved.AOP介绍AOP提供了另外一种思考程序结构的角度,弥补了OOP(面向对象编程)的不足。就像刚开始理解OO概念一样,对于新手来说AOP也是非常抽象难以理解的,不能仅从一个概念上去定义AOP。假如我们有一个系统,分为好多个模块,每个模块都负责处理一项重要的功能。但是每个模块都需要一些相似的辅助功能如安全、日志输出等等。这就是一种交叉业务,而这种“交叉”非常适合用AOP来解决。4©2007BrillianceTechnologyAllrightsreserved.AOP术语切面(Aspect):切面是你要实现的交叉功能。它是应用系统模块化的一个切面或领域。连接点(Joinpoint):连接点是应用程序执行过程中插入切面的地点(在程序执行过程中某个特定的点)。在SpringAOP中一个连接点代表一个方法的执行。这个点可以是方法调用,异常抛出或者是要修改的字段。通过申明一个importorg.aspectj.lang.JoinPoint类型的参数可以使通知(Advice)的主体部分获得连接点信息。通知(Advice):通知切面的实际实现。它通知应用系统新的行为。通知包括好多种类,在后面单独列出。切入点(Pointcut):切入点定义了通知应该应用在哪些连接点。通知(Advice)可以应用到AOP的任何连接点。通知(Advice)将和一个切入点表达式关联,并在满足这个连接点的切入点上运行(例如:在执行一个特定名称的方法时)切入点表达式如何和连接点匹配是AOP的核心Spring使用缺省的AspectJ切入点的语法。引入(Introduction):引入允许你为已经存在的类添加新的方法和属性。Spring允许引入新的接口(以及一个对应的实现)到任何被代理的对象。目标对象(TargetObject):目标对象是被通知对象。SpringAOP是运行时代里实现的,所以这个对象永远是一个被代理对象。5©2007BrillianceTechnologyAllrightsreserved.AOP代理(AOPProxy):代理是将通知(Advice)应用到目标对象后创建的对象。织入(Weaving):把切面(Aspect)连接到其它的应用程序类型或者对象上来创建一个被通知(advised)的对象。可以在编译时做这件事(例如使用AspectJ编译器),也可以在类加载或运行时完成。Spring和其他纯JavaAOP框架一样,在运行时完成织入。通知(Advice)类型:前置通知(BeforeAdvice):在一个连接点之前执行的通知,但这个通知不能阻止连接点前的执行(除非它抛出异常)返回后通知(Afterreturningadvice):在一个连接点正常完成后执行的通知。例如:一个方法正常返回,没有抛出任何异常。抛出后通知(Afterthrowingadvice):在一个方法抛出异常时执行的通知。Finally后通知(Afterfinallyadvice):当某连接点退出的时候执行的通知(不论是正常返回还是抛出异常)。环绕通知(Aroundadvice):包围一个连接点的通知,就像方法调用。这是最强大的一种通知类型。环绕通知可以在方法调用前后完成自定义的行为。它也会选择是否继续执行连接点或直接返回他们自己的返回值域或抛出异常来结束执行。6©2007BrillianceTechnologyAllrightsreserved.议题AOP介绍创建通知定义切入点使用ProxyFactoryBean自动代理7©2007BrillianceTechnologyAllrightsreserved.1.前置通知(BeforeAdvice)创建前置通知需要实现org.springframework.aop.MethodBeforeAdvice接口,该接口只有before(Methodmethod,Object[]args,Objecttarget)throwsThrowable这个方法。我们可以这样理解前置通知:比如你到一个比较高档的饭店去吃饭,进门时会有礼仪小姐给你开门并且向您问好,这个礼仪小姐的行为就可以视为“前置通知”。packagespringinaction.chapter03.createadvice;importjava.lang.reflect.Method;importorg.springframework.aop.MethodBeforeAdvice;publicclassBeforeAdviceimplementsMethodBeforeAdvice{//实现MethodBeforeAdvice的before方法publicvoidbefore(Methodmethod,Object[]args,Objecttarget){System.out.println(beforeadvice);}//endbefore}//endclassBeforeAdvice8©2007BrillianceTechnologyAllrightsreserved.2.后置通知(Afterreturningadvice)创建返回后通知需要实现org.springframework.aop.AfterReturningAdvice接口,该接口也只有一个方法:voidafterReturning(ObjectreturnValue,Methodmethod,Object[]args,Objecttarget)throwsThrowable。同样我们也可以这样理解Afterreturningadvice:当你在这家饭店用餐完后,礼仪小姐还会为您开门,并且欢迎您下次再来。这时礼仪小姐的行为就是“Afterreturningadvice”。packagespringinaction.chapter03.createadvice;importjava.lang.reflect.Method;importorg.springframework.aop.AfterReturningAdvice;publicclassAfterReturningAdviceImpimplementsAfterReturningAdvice{publicvoidafterReturning(ObjectreturnValue,Methodmethod,Object[]arg2,Objecttarget)throwsThrowable{System.out.println(afterReturning);}}9©2007BrillianceTechnologyAllrightsreserved.3.环绕通知(Aroundadvice)创建环绕通知需要实现org.aopalliance.intercept.MethodInterceptor接口,同样要实现一个invoke方法,该方法有一个MethodInvocation类型的参数。MethodInterceptor能够控制目标方法是否真的被调用。通过调用MethodInvocation.proceed()方法来调用目标方法。MethodInterceptor让你可以控制返回的对象,你可以返回一个与proceed()方法返回对象完全不同的对象。我们可以这样理解环绕通知:当你用餐完后,饭店要确保你已经付款了,在付款后停止再出售食物给你。packagespringinaction.chapter03.createadvice;importorg.aopalliance.intercept.MethodInterceptor;importorg.aopalliance.intercept.MethodInvocation;publicclassAroundAdviceimplementsMethodInterceptor{publicObjectinvoke(MethodInvocationmi){Objectobj=null;//dosomething....returnobj;}//endinvoke(...)}//endclassAroundAdvice()10©2007BrillianceTechnologyAllrightsreserved.4.异常通知(Afterthrowingadvice)创建抛出后通知需要实现org.springframework.aop.ThrowsAdvice接口,该接口没有任何方法,但是要实现这个接口的类必须实现afterThorwing(Throwablethrowable)或者afterThrowing(Methodmethod,Object[]args,Objecttarget)形式的一种,根据抛出的异常的类型恰当的方法将被调用。如果你在该饭店用餐后没有付钱就走或者多给钱他们没有找给你大概异常通知也就发生了。packagespringinaction.chapter03.createadvice;importjava.lang.reflect.Method;importorg.springframework.aop.ThrowsAdvice;publicclassAfterThorwsAdviceimplementsThrowsAdvice{publicvoidafterThorwing(Throwablethrowable){//dosomething....}publicvoidafterThrowing(Methodmethod,Object[]args,Objecttarget){//dosomething....}}一般来说环绕通知是用的最广泛的一个通知类型,但是他们(Spring小组)鼓励我们用最合适的通知。例如我们有一个简单的验证身份的功能,那么我们只需要前置通知就可能实现我们要求的功能了。11©2007BrillianceTechnologyAllrightsreserved.议题AOP介绍创建通知定义切入点使用ProxyFactoryBean自动代理12©2007BrillianceTechnologyAllrightsreserved.定义切入点在上面我们定义了各种通知。可以看到一个通知是要被执行的一个方法。光把通知定义出来不行,我们还要确定这些通知在我们的系统什么地方应用,否则通知是毫无用处的。这就是切入点的用处。切入点决定了一个特定的类的特定方法是否满足一条特定规则。如果确实符合,通知就应用到该方法上。Spring根据需要织入通知的类和方法来定义切入点。Advice根据他们的特性织入目标类和方法。Spring的切入点框架的核心接口是Pointcut。PublicinterfacePointcut{ClassFiltergetClassFilter();MethodMatchergetMethodMatcher();}13©2007BrillianceTechnologyAllrightsreserved.ClassFilter接口决定了一个类是否符合通知的要求:packageorg.springframework.aop;publicinterfaceClassFilter{booleanmatches(Classclass);}实现这个接口的类决定了以参数传入进来的类是否应该被通知。ClassFilter.TRUE是规范的的适合任何类的ClassFilter实例,他适用于创建只根据方法决定时候符合要求的切入点。ClassFilter接口利用类过滤切面,MethodMactcher接口可以通过方法过滤切面:packageorg.spring