在编写 AspectJ 切面时, 可以直接在通知注解中书写切入点表达式. 但同一个切点表达式可能会在多个通知中重复出现。在 AspectJ 切面中, 可以通过 @Pointcut 注解将一个切入点声明成简单的方法. 切入点的方法体通常是空的, 因为将切入点定义与应用程序逻辑混在一起是不合理的。切入点方法的访问控制符同时也控制着这个切入点的可见性. 如果切入点要在多个切面中共用, 最好将它们集中在一个公共的类中. 在这种情况下, 它们必须被声明为 public. 在引入这个切入点时, 必须将类名也包括在内. 如果类没有与这个切面放在同一个包中, 还必须包含包名。其他通知可以通过方法名称引入该切入点。
@Aspect @Component public class HelloAspect { @Pointcut("execution(public void com.aop.HelloImpl.hi())") private void operate() { } // 声明该方法为前置通知 @Before("operate()") public void beforeMethod(JoinPoint point) { String methodName = point.getSignature().getName(); System.out.println("HelloAspect beforeMethod : " + methodName); } }
3 基于XML的配置声明切面除了使用 AspectJ 注解声明切面, Spring 也支持在 Bean 配置文件中声明切面. 这种声明是通过 aop schema 中的 XML 元素完成的。正常情况下, 基于注解的声明要优先于基于 XML 的声明. 通过 AspectJ 注解, 切面可以与 AspectJ 兼容, 而基于 XML 的配置则是 Spring 专有的. 由于 AspectJ 得到越来越多的 AOP 框架支持, 所以以注解风格编写的切面将会有更多重用的机会。
3.1 声明切面XML中声明切面时,需要在<beans>根元素中导入aop schema,在Bean配置文件中,所有的Spring AOP配置都必须定义在<aop:config>元素内部,对于每个切面来说,都要创建一个<aop:aspect>元素来为具体的切面实现引用后端Bean实例。
<aop:config> <aop:aspect ref="helloAspect"> <aop:before pointcut="execution(public void com.aop.HelloImpl.*())" method="beforeMethod"/> </aop:aspect> <aop:aspect ref="helloAspect2"> <aop:before pointcut="execution(public void com.aop.HelloImpl.*())" method="beforeMethod"/> </aop:aspect> </aop:config>
3.2 声明切入点和通知切入点使用<aop:pointcut>元素声明,切入点必须定义在<aop:sapect>元素下,或者直接定义在<aop:config>元素下。
定义在 <aop:aspect> 元素下: 只对当前切面有效
定义在 <aop:config> 元素下: 对所有切面都有效
通知元素使用<pointcut-ref>来引用切入点,或直接把<pointcut>嵌入到切入点表达式中,method属性指定切面类中通知方法的名称。
<aop:config> <aop:pointcut id="pointcut" expression="execution(public void com.aop.HelloImpl.*())"/> <aop:aspect ref="helloAspect"> <aop:before pointcut-ref="pointcut" method="beforeMethod"/> </aop:aspect> <aop:aspect ref="helloAspect2"> <aop:before pointcut-ref="pointcut" method="beforeMethod"/> </aop:aspect> </aop:config>
基于 Spring 设计并实现 RESTful Web Services