SpringAop使用到了了动态代理模式(有关设计模式见设计模式章节)。JDK动态代理代理的目标类必须要实现接口。因此当目标类没有实现接口时,会使用CGLIB来为目标对象创建代理实例【原理是对目标类进行继承扩展,为其生成相应的子类,然后将横切逻辑写到子类当中】。
要了解AOP,首先要了解几个关于它的术语:
①join point:连接点,程序运行时的一些时间点,如方法执行点,异常处理点。
②point cut:切点,在SpringAop中,所有方法执行都是连接点,但并不是所有连接点都需要执行advice,切点就是用来描述哪些连接点来执行advice
③advice:增强,即需要新增的的部分,advice与point cut匹配 ,advice类型有以下几种:
before advice:在join point前被执行的advice
after return advice:在一个join point 正常返回后执行的advice
after throw advice:在一个join point异常返回后执行的advice
after advice
④aspect:切面,即point cut和advice总称, 它既包含了横切逻辑的定义, 也包括了连接点的定义
⑤target:目标对象,即织入切面后的对象
基本使用:
@Aspect注解:使用aspect注解必须需要aspectjweaver.jar包,并且通过xml或configutration方式开启使用
定义切面:要注意两个地方 使用@Aspect注解只是声明一个切面,还需使用@Componet将其注入到Spring 容器中, 如果一个 类被@Aspect 标注, 则这个类就不能是其他 aspect 的 target 了, 因为使用 @Aspect 后, 这个类就会被排除在 auto-proxying 机制之外.【包括@Transactional ,例子:】
声明切点:一个切点有两部分组成:方法签名【方法名和方法参数】、匹配表达式
定义advice
当然,也可以将切点和增强一同定义如下:【此时无需声明切点】
这里可以通过joinPonit来获取原方法的相关信息,常用的api如下:
使用环绕advice时,有几点不同
1、目标方法的调用由环绕通知决定,即你可以决定是否调用目标方法,而前置和后置通知 不能决定的,他们只是在方法的调用前后执行通知而已,即目标方法肯定是要执行的。
2、环绕通知可以控制返回对象,即你可以返回一个与目标对象完全不同的返回值,虽然这很危险,但是你却可以办到。而后置方法是无法办到的,因为他是在目标方法返回值后调用【即proced返回值】
实际使用:
aop应用的地方有很多,如验证码检查,简单的日志管理等等,如我们项目通过aop做图形验证码检查【业务逻辑太多代码就不贴了。和前面差不多,只是注解带着需要检验的值,如果有兴趣可以交流】,注意一下获取所需参数,及proceed方法抛出异常时reponse生成