Spring aop 面向切面编程,前不久由于需要,对返回值类型相同的方法添加缓存。由于这些方法分散在不同的地方,所以需要使用spring aop编程。aop实现的过程就是在什么地点做什么事情。
首先,先要了解几个概念
- 横切关注点
对哪些方法进行拦截,拦截以后怎么处理,这些关注点被成为横切关注点。 - 切面(Aspect)
类是对物体特征的抽象,切面是对横切关注点的抽象。 - 连接点(JoinPoint)
被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器. - 切入点(pointcut)
对连接点进行拦截的定义 - 通知(advice)
所谓通知指的就是指拦截到连接点之后要执行的代码,通知分为前置、后置、异常、最终、环绕通知五类 - 目标对象
代理的目标对象 - 织入(weave)
将切面应用到目标对象并导致代理对象创建的过程 - 引入(introduction)
在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段.
Spring aop 代理方式
Spring中Aop代理有Spring的IOC容器负责生成,管理,其依赖关系也有IOC容器负责管理。因此,AOP代理可以直接使用容器中的其他bean实例作为目标,中中管理可以IOC容器依赖注入提供。Spring 创建代理的规则是:
- 默认使用java动态代理来创建AOP代理,这样就可以为任何借口实例创建代理。
- 当需要代理的类不是代理接口的时候,Spring会切换为使用CGLIB代理,也可以强制使用CGLIB。
AOP编程的三个步骤
- 定义普通的业务组件
- 定义切入点,一个切入点可能横切多个业务组件
- 定义增强处理,增强处理就是AOP框架为不同业务织入的动作。
即:代理对象的方法 = 增强代理 + 被代理对象的方法
下面是我做测试使用的小例子,如果有不对的地方请多指正。
我是采用注解的方式进行进行aop编程的。
首先定义一个注解
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TestAnno{
String methodName() default "";
}
然后定义一个切面,切面首先是一个组件,然后添加注解@Aspect声明这是一个切面类。在组件内部 可以有各种通知。通过doBefore,Around等注解来声明。下面采用的@Around 环绕通知。@Around可以有返回值。
@Component
@Aspect
public class TestAspect{
public Object aroundTest(ProceedingJoinPoint point){
//当执行point.proceed()以后 切入的方法不会被增强处理。
Object result = point.proceed();
return result;
}
}
然后需要在配置文件中配置。aop的配置可以在context.xml中配置,也可以在spring-mvc.xml中配置。由于spring 扫描的顺序不同,配置在context中的aop只会在Service层生效。而配置在spring-mvc.xml的aop会在Controller层生效。
在配置文件中添加 spring的命名空间
xmlns:aop="http://www.springframework.org/schema/aop"
并在xsi:schemaLocation的后面添加如下配置:
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
然后需要的类配置在spring 中(包括注解,被代理的方法所在的类以及切面类),在这里采用自动扫描包的形式。并启动Aspect支持
<context:component-scan baspackage="com.xxx.maven.demo.*" />
<!-- 启动@AspectJ支持 -->
<aop:aspectj-autoproxy proxy-target-class="true" >
<!-- <aop:include name="dataSourceAspect" /> -->
</aop:aspectj-autoproxy>
使用spring aop除了使用spring的jar以外,还需要以下依赖
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.7.4</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjrt</artifactId>
<version>1.7.4</version>
</dependency>