四. 使用@EnableAspectJAutoProxy注解启用自动代理功能, 如果是XML Config ,对应的节点是<aop:aspectj-autoproxy />
@Configuration @ComponentScan // 包扫描 @EnableAspenctJAutoProxy // 启动自动代理 public class MyConfig{ // 如果Audience上加了@Component就不需要这个代码了 @Bean public Audience getAudience(){ return new Audience(); } }五. 使用环绕通知@Around, 环绕通知同时兼具了@Before,@After ... 等注解的方法的功能, 下面代码演示了这种能力. 如可以使用它记录方法执行时长.
@Aspect public class Audience{ //定义切点并修改其他方法重用该切点 @Pointcut("execution(** com.xlx.Performance.perform(...))") public void performance(){ } @Around("performance()") public void silencephone(ProcdedingJoinPoint jp){ System.out.println("silencephone"); System.out.println("takeSeats"); try{ // 如果不是刻意为之, 一定要记得调用jp.proceed();否则实际的方法Performance.perform()将会阻塞 jp.proceed(); System.out.println("applause"); }catch(Exception e){ System.out.println("refund"); } } }六. 参数传递 , 在切点表达式中使用args(paramName)结合切点方法可以为切面方法传递参数
@Aspect public class Audience{ //定义切点并修改其他方法重用该切点 @Pointcut("execution(** com.xlx.Performance.perform(int) && args(actornum)))") public void performance(int actornum){ } @Before("performance(actornum)") public void countActor(int actornum){ System.out.println("countActor"+actornum); } } 通过注解引用新功能除了拦截对象已有的方法调用, 还可以使用AOP来为对象添加新的属性和行为(引入). 其实现就是通过动态代理生成代理类来实现.
一. 定义要添加的功能接口
public interface Encoreable{}二. 定义切面(引入) @Aspect注解切面类. @DeclareParents注解功能接口静态变量
@Aspect public class EncoreableIntroducer{ // 可以解释为: 为Performace的所有子类引入接口Encoreable, 并使用默认实现类DefaultEncoreableImpl @DeclareParents(value="xlx.Performace+",defaultImpl=DefaultEncoreableImpl.class) public static Encoreable encoreable; } 基于XML配置的切面如果没有办法为类添加注解, 比如没有源代码, 那就不得不使用xml来配置了.
示例1
<aop:config> <aop:aspect ref="aspectBean"> <aop:pointcut expression="execution(** com.xlx.Performance.perform(int) and args(actornum)))" /> <aop:before pointcut-ref="pcId" method="count" /> </aop:aspect> </aop:config>示例2
<aop:config> <aop:aspect> <aop:declare-parents type-matching="xlx.Performace+" implement-interface="xlx.Encoreable" delegate-ref="defaultImpl" /> </aop:aspect> </aop:config> AspectJ 注入使用AspectJ注入的方式可以解决使用动态代理无法解决的问题(应该比较少见,大多应用使用Spring AOP就可以实现了), 但需要使用AspectJ的特殊语法. 定义好的类需要用xml配置为bean, 使用factory-method="aspectOf"属性来制定bean的产生方式.
<bean factory-method="aspectOf"> <property ref="otherref"/> </bean>