头图

Hello,今天给各位童鞋们分享Spring AOP,赶紧拿出小本子记下来吧!
image.png

概述

AOP是aspect-oriented programming的缩写,译为面向切面编程。利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得 业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。简单来说,AOP就是不修改源代码在主干功能里面添加新功能。

底层原理‍‍

AOP底层使用了动态代理:在有接口的时候使用JDK 动态代理、在没有接口的时候使用CGLIB字节码动态代理。

JDK动态代理

简介

使用JDK 动态代理需要用到JDK中的Proxy类里面的newProxyInstance方法创建代理对象。方法如下:

static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h);

newProxyInstance方法的三个参数说明:

loader 定义代理类的类加载器

interfaces 代理类要实现的接口列表(可以多个)

h 指派方法调用的处理程序(要增强的功能在这里面实现)

newProxyInstance方法返回一个指定接口的代理类实例。

InvocationHandler接口中有个invoke方法,用于实现增强的功能:

public Object invoke(Object proxy, Method method, Object[] args)

    throws Throwable;

invoke方法的三个参数说明:

proxy表示代理对象

method表示被增强的方法

args是方法的参数 若没有则为null

示例代码
代码结构如下:

└─src

└─com

    └─spring5

            JDKProxy.java

            UserDao.java

            UserDaoImpl.java

UserDao接口:

public interface UserDao {

 int add(int a,int b);

 String update(String id);

}

UserDaoImpl类:
image.png

JDKProxy类:
image.png

运行结果:

方法之前执行…add :传递的参数…[1, 2]

I am add…

方法之后执行…com.spring5.UserDaoImpl@355da254

result:3

CGLIB字节码动态代理

简介

使用CGLIB字节码动态代理不受代理类必须实现接口的限制,其底层采用ASM字节码生成框架。CGLIB动态代理的优缺点:

使用字节码技术生产代理类比JAVA反射效率高

不能对声明为final的方法进行代理,因为其原理是动态生成被代理类的子类

需要实现接口MethodInterceptor,然后重写intercept方法:

Object intercept(Object proxy, Method method, Object[] args, MethodProxy arg3) throws Throwable;

intercept方法的参数说明:

  • proxy CGLIB生成的代理类实例,也是目标对象的子类,相当于重写父类方法
  • method 被代理方法
  • args 方法参数
  • 为生成的代理类对方法的代理引用

intercept方法返回

另外用到了Enhancer类,它是Cglib中的一个字节码增强器,先调它的setSuperclass()将被代理类设置成父类、再调setCallback函数执行intercept方法,最后调create()生成代理类。

示例代码

代码结构如下:

└─src

└─com

    └─spring5

            CglibProxy.java

            User.java

User类:
public class User {

public void sleep() {

    System.out.println("我想睡觉...");

}

}

CglibProxy类:
image.png

运行结果:

睡觉前脱衣服

我想睡觉…

起床穿衣服

​AOP操作

概述

AOP相关的几个术语:

  • 连接点

类里面哪些方法可以被增强,这些方法称为连接点

  • 切入点

实际被真正增强的方法称为切入点

  • 通知

实际增强的逻辑部分称为通知,分为前置通知、后置通知、环绕通知、异常通知和最终通知五种类型,其中最终通知相当于JAVA的finally。

  • 切面

把通知应用到切入点过程

AspectJ

AspectJ不是 Spring 组成部分,是一个独立的AOP 框架,一般把 AspectJ 和 Spirng 框架一起使 用,进行 AOP操作。增强就是代理的意思。

准备工作
在进行AOP操作的时候需要先引入下面四个Jar包

com.springsource.net.sf.cglib-2.2.0.jar

com.springsource.org.aopalliance-1.0.0.jar

com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar

spring-aspects-5.2.6.RELEASE.jar

引入包后的所有包如下:

com.springsource.net.sf.cglib-2.2.0.jar

  • com.springsource.org.aopalliance-1.0.0.jar
  • com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
  • spring-aspects-5.2.6.RELEASE.jar
  • commons-logging-1.1.1.jar
  • spring-aop-5.2.6.RELEASE.jar
  • spring-beans-5.2.6.RELEASE.jar
  • spring-context-5.2.6.RELEASE.jar
  • spring-core-5.2.6.RELEASE.jar
  • spring-expression-5.2.6.RELEASE.jar

AspectJ的切入点表达式相关说明如下:

  • 作用

知道对哪个类里面的哪个方法进行增强

  • 语法

execution([权限修饰符] [返回类型] [类全路径] 方法名称 )

  • 示例
    image.png

基于注解实现

主要步骤

主要步骤如下:

在spring 配置文件中,开启注解扫描

需要在XML中引入context和aop的上下文空间。

使用注解创建 User 和 UserProxy 对象

在增强类上面添加注解 @Aspect

在 spring配置文件中开启生成代理对象

<aop:aspectj-autoproxy></aop:aspectj-autoproxy>

开启Aspectj生成对象后,会去代码中扫描@aspect注解

配置不同类型的通知

在通知方法上面使用@Before、@AfterReturning、@Around、@AfterThrowing和@After注解,结合切入点表达式配置。

@after是在方法执行之后执行(有异常也执行),@afterReturning是在返回值之后执行(有异常不执行)。

示例代码
代码结构:
image.png

User类:
image.png

UserProxy类:
image.png

Test类:
image.png

无异常返回结果:

Around 环绕之前

前置通知 before

I am add

Around 环绕之后

最终通知 after

后置通知(返回通知)afterReturning

有异常时返回结果:
Around 环绕之前
前置通知 before

最终通知 after

异常通知 afterThrowing

java.lang.ArithmeticException: / by zero

相同切入点提取
用@Pointcut标签
image.png

多个增强类对同一个方法进行增强
用@Order注解设置增强类优先级,数字类型值越小表示优先级越高。

@Component

@Aspect

@Order(1)

public class PersonProxy{ }

完全注解开发
在启动配置类中添加@EnableAspectJAutoProxy注解:

@Configuration

@ComponentScan(basePackages = { "com.spring5"})

@EnableAspectJAutoProxy(proxyTargetClass = true)

public class ConfigAop {

}

基于配置文件实现
具体步骤
创建增强类和被增强类,创建相关方法

在spring 配置文件中配置两个类对象

在 spring 配置文件中配置AOP

示例代码
代码结构如下:
image.png

Student类:
image.png

StudentProxy类:
image.png

Test类:
image.png

bean.xml:
image.png

运行结果:

I am before…

I want to buy a book…

I am afterReturn…

好啦,今天的文章就到这里,希望能帮助到屏幕前迷茫的你们


java小孩
25 声望0 粉丝

定期普及、分享java方面的知识