关键字: pointcut的定义
在
Spring 2.0
中,
Pointcut
的定义包括两个部分:
Pointcut
表示式
(expression)
和
Pointcut
签名
(signature)
。让我们先看看
execution
表示式的格式:
括号中各个 pattern 分别表示修饰符匹配( modifier-pattern? )、返回值匹配( ret-type-pattern )、类路径匹配( declaring-type-pattern? )、方法名匹配( name-pattern )、参数匹配( (param-pattern) )、异常类型匹配( throws-pattern? ),其中后面跟着“ ? ”的是可选项。
在各个 pattern 中可以使用“ * ”来表示匹配所有。在 (param-pattern) 中,可以指定具体的参数类型,多个参数间用“ , ”隔开,各个也可以用“ * ”来表示匹配任意类型的参数,如 (String) 表示匹配一个 String 参数的方法; (*,String) 表示匹配有两个参数的方法,第一个参数可以是任意类型,而第二个参数是 String 类型;可以用 (..) 表示零个或多个任意参数。
现在来看看几个例子:
1 ) execution(* *(..))
表示匹配所有方法
2 ) execution(public * com. savage.service.UserService.*(..))
表示匹配 com.savage.server.UserService 中所有的公有方法
3 ) execution(* com.savage.server..*.*(..))
表示匹配 com.savage.server 包及其子包下的所有方法
除了 execution 表示式外,还有 within 、 this 、 target 、 args 等 Pointcut 表示式。 一个 Pointcut 定义由 Pointcut 表示式和 Pointcut 签名组成,例如:
java 代码
- execution(modifier-pattern?
- ret-type-pattern
- declaring-type-pattern?
- name-pattern(param-pattern)
- throws-pattern?)
括号中各个 pattern 分别表示修饰符匹配( modifier-pattern? )、返回值匹配( ret-type-pattern )、类路径匹配( declaring-type-pattern? )、方法名匹配( name-pattern )、参数匹配( (param-pattern) )、异常类型匹配( throws-pattern? ),其中后面跟着“ ? ”的是可选项。
在各个 pattern 中可以使用“ * ”来表示匹配所有。在 (param-pattern) 中,可以指定具体的参数类型,多个参数间用“ , ”隔开,各个也可以用“ * ”来表示匹配任意类型的参数,如 (String) 表示匹配一个 String 参数的方法; (*,String) 表示匹配有两个参数的方法,第一个参数可以是任意类型,而第二个参数是 String 类型;可以用 (..) 表示零个或多个任意参数。
现在来看看几个例子:
1 ) execution(* *(..))
表示匹配所有方法
2 ) execution(public * com. savage.service.UserService.*(..))
表示匹配 com.savage.server.UserService 中所有的公有方法
3 ) execution(* com.savage.server..*.*(..))
表示匹配 com.savage.server 包及其子包下的所有方法
除了 execution 表示式外,还有 within 、 this 、 target 、 args 等 Pointcut 表示式。 一个 Pointcut 定义由 Pointcut 表示式和 Pointcut 签名组成,例如:
java 代码
- //Pointcut表示式
- @Pointcut("execution(* com.savage.aop.MessageSender.*(..))")
- //Point签名
- private void log(){} 然后要使用所定义的Pointcut时,可以指定Pointcut签名,如
java 代码- @Before("og()")
上面的定义等同与:
java 代码- @Before("execution(* com.savage.aop.MessageSender.*(..))")
Pointcut定义时,还可以使用&&、||、!运算,如:
java 代码- @Pointcut("execution(* com.savage.aop.MessageSender.*(..))")
- private void logSender(){}
- @Pointcut("execution(* com.savage.aop.MessageReceiver.*(..))")
- private void logReceiver(){}
- @Pointcut("logSender() || logReceiver()")
- private void logMessage(){}
这个例子中,logMessage()将匹配任何MessageSender和MessageReceiver中的任何方法。
还可以将一些公用的Pointcut放到一个类中,以供整个应用程序使用,如:
java 代码- package com.savage.aop;
- import org.aspectj.lang.annotation.*;
- public class Pointcuts {
- @Pointcut("execution(* *Message(..))")
- public void logMessage(){}
- @Pointcut("execution(* *Attachment(..))")
- public void logAttachment(){}
- @Pointcut("execution(* *Service.*(..))")
- public void auth(){}
- }
在使用这些Pointcut时,指定完整的类名加上Pointcut签名就可以了,如:
java 代码- package com.savage.aop;
- import org.aspectj.lang.JoinPoint;
- import org.aspectj.lang.annotation.*;
- @Aspect
- public class LogBeforeAdvice {
- @Before("com.sagage.aop.Pointcuts.logMessage()")
- public void before(JoinPoint joinPoint) {
- System.out.println("Logging before " + joinPoint.getSignature().getName());
- }
- }
当基于XML Sechma实现Advice时,如果Pointcut需要被重用,可以使用<aop:pointcut></aop:pointcut>来声明Pointcut,然后在需要使用这个Pointcut的地方,用pointcut-ref引用就行了,如:
xml 代码- <aop:config>
- <aop:pointcut id="log"
- expression="execution(* com.savage.simplespring.bean.MessageSender.*(..))"/>
- <aop:aspect id="logging" ref="logBeforeAdvice">
- <aop:before pointcut-ref="log" method="before"/>
- <aop:after-returning pointcut-ref="log" method="afterReturning"/>
- </aop:aspect>
- </aop:config>