Spring:AOP

来自Wikioe
跳到导航 跳到搜索


关于AOP

Aspect Oriented Programming:面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的。

  • AOP是OOP的延续。
  • 可以在不修改源码的情况下,对程序进行增强;
  • 应用如:权限校验,日志记录,性能监控,事务控制;

AOP 实现

Spring 的AOP 的底层用到两种代理机制:

  • JDK 动态代理:(实现了接口)使用动态代理创建接口实现类代理对象
  • CGLIB 动态代理:(没有实现接口)使用动态代理创建类的子类代理对象;(在子类中调用父类的方法,来完成增强)
    • 应用的是底层的字节码增强的技术生成当前类的子类对象;

JDK 动态代理

public class MyJDKProxy implements InvocationHandler {
	private UserDao userDao;
	
	public MyJDKProxy(UserDao userDao) {
		this.userDao = userDao;
	}
	
	// 编写工具方法:生成代理:
	public UserDao createProxy() {
		UserDao userDaoProxy = (UserDao)Proxy.newProxyInstance(userDao.getClass().getClassLoader(),
																userDao.getClass().getInterfaces(), this);
		return userDaoProxy;
	}
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		if("save".equals(method.getName())) {
			System.out.println("权限校验================");
		}
		return method.invoke(userDao, args);
	}
}

CGLIB 动态代理

public class MyCglibProxy implements MethodInterceptor {
	private CustomerDao customerDao;
	
	public MyCglibProxy(CustomerDao customerDao) {
		this.customerDao = customerDao;
	}
	
	// 生成代理的方法:
	public CustomerDao createProxy() {
		// 创建Cglib 的核心类:
		Enhancer enhancer = new Enhancer();
		// 设置父类:
		enhancer.setSuperclass(CustomerDao.class);
		// 设置回调:
		enhancer.setCallback(this);
		// 生成代理:
		CustomerDao customerDaoProxy = (CustomerDao) enhancer.create();
		return customerDaoProxy;
	}
	
	@Override
	public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
		if("delete".equals(method.getName())) {
			Object obj = methodProxy.invokeSuper(proxy, args);
			System.out.println("日志记录================");
			return obj;
		}
		return methodProxy.invokeSuper(proxy, args);
	}
}

关于AspectJ

AspectJ,(Eclipse AspectJ)是一个易用的功能强大的AOP框架;可以单独使用(需要专门的编译器“ajc”),也可以整合到其它框架中。

  • 官网:“http://www.eclipse.org/aspectj/”;
  • AspectJ是AOP编程的完全解决方案,可以做Spring AOP干不了的事情;
  • AspectJ不是spring一部分,和spring一起使用进行aop操作(Spring2.0以后新增了对AspectJ支持);

AspectJ 与 Spring AOP

AOP编程,存在三种织入方式:

  1. 编译期织入(Compile-time weaving):(编译期、编译后)在Java编译期,采用特殊的编译器,将切面织入到Java类中;
    • 编译后:对已经生成的“.class”文件、“jar”包进行织入;
  2. 类加载期织入(Load-time weaving):通过特殊的类加载器,在类字节码加载到JVM时,织入切面;
    • 或在JVM启动的时候指定 AspectJ 提供的“agent:-javaagent:xxx/xxx/aspectjweaver.jar”;【?】
  3. 运行期织入:采用CGLib工具JDK动态代理进行切面的织入;


Spring AOP 和 AspectJ 之间的关键区别:

AspectJ Spring AOP
静态织入
  • 支持编译时、编译后和加载时织入
运行期织入
支持所有切入点
  • (即:可以编织字段、方法、构造函数、静态初始值设定项、最终类/方法等编织)
仅支持方法执行切入点
  • (即:仅支持方法级编织)
使用 Java 编程语言的扩展实现
  • 除非设置LTW(Load Time Weaving,加载时织入),否则需要 AspectJ 编译器 (ajc)
在纯 Java 中实现
  • (不需要单独的编译过程)
更好的性能 较 AspectJ 慢
  • Spring AOP使用了Aspect的Annotation,但是并没有使用它的编译器和织入器;
    “@Before”、“@After”等是Aspect的Annotation;
  • AspectJ在实际运行之前就完成了织入,所以其生成的类没有额外运行时开销。

相关术语

  • Joinpoint(连接点):所谓连接点是指那些被拦截到的点。【可以被切入的方法】
    • 在spring 中,这些点指的是方法,因为spring 只支持方法类型的连接点;
  • Pointcut(切入点):所谓切入点是指我们要对哪些Joinpoint 进行拦截的定义;【用来切入的方法】
  • Advice(通知/增强):所谓通知是指拦截到Joinpoint 之后所要做的事情就是通知;【切面处理】
    1. “前置通知”:;
    2. “后置通知”:;
    3. “异常通知”:;
    4. “最终通知”:无论程序是否正常执行,最终通知的代码会得到执行;
    5. “环绕通知”:;
  • Introduction(引介):引介是一种特殊的通知在不修改类代码的前提下, Introduction可以在运行期为类动态地添加一些方法或Field;
  • Target(目标对象):代理的目标对象;
  • Weaving(织入):是指把增强应用到目标对象来创建新的代理对象的过程;【应用增强生成代理对象】
    • spring采用“动态代理织入”,而AspectJ采用“编译期织入”和“类装载期织入”;【?】
  • Proxy(代理):一个类被AOP 织入增强后,就产生一个结果代理类;
  • Aspect(切面): 是切入点和通知(引介)的结合;

jar包

  1. spring 的传统AOP 的开发的包:
    • “spring-aop-4.2.4.RELEASE.jar”
    • “com.springsource.org.aopalliance-1.0.0.jar”
  • aspectJ 的开发包:
    • “com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar”
    • “spring-aspects-4.2.4.RELEASE.jar”

配置文件

需要在“applicationContext.xml”中引入AOP相关约束:

<beans xmlns="http://www.springframework.org/schema/beans"
   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
   xmlns:aop="http://www.springframework.org/schema/aop" 
   xsi:schemaLocation="http://www.springframework.org/schema/beans
            http://www.springframework.org/schema/beans/spring-beans.xsd
               http://www.springframework.org/schema/aop
            http://www.springframework.org/schema/aop/spring-aop.xsd">
</beans>

基于AspectJ:XML配置

基于AspectJ:注解