Spring5系列(十一) | 基于注解的AOP编程

2021-12-13 09:44:09 浏览数 (1)

概述: 本篇文章很重要! 工作中我们经常会遇到给我们的项目写一个切面,很多开发工程师刚开始的时候都不知道切面应该怎么写,本篇文章就会教大家如何开发一个切面。

我们前面讲解了Spring的AOP编程,本质就是给spring的对象通过创建代理对象的方式添加额外功能。我们前面的方式都是通过在xml配置的方式实现的。我们简单回顾一下之前的步骤。

  1. 原始对象
  2. 额外功能
  3. 切入点
  4. 组装

一、 开发步骤

代码语言:javascript复制
1. 额外功能:之前写法
			public class MyArround implements MythodInterceptor{
  public Object  invoke(MethodInvocation invocation){...}
  
  }
2. 切入点: 之前写法
<aop:confg>
	<aop:pointcut id="pc" expression="executionn(* com.xxx..*.*(..))" />
    <aop:advisor advice-ref="around" pointcut-ref="pc" />
</aop:confg> 
复制代码

Spring本身为我们提供了注解的方式,来实现AOP的编程,我们来看下代码.

  1. 创建切面类,通过切面类定义额外功能和切入点。
代码语言:javascript复制
/**
1. 额外功能:之前写法
      public class MyArround implements MythodInterceptor{
          public Object  invoke(MethodInvocation invocation){}
     }
2. 切入点: 之前写法
    <aop: config>
        <aop:pointcut id="pc" expression="execution(* login(..))" />
    </aop:config>
*/
@Aspect // 指定是切入类
public class MyAspect{
	
    @Arround("execution(* login(..))") // 指定额外功能和切入点表达式
    public Object arround(ProceedingJoinPoint joinPoint) throws Throwable{
    System.out.println("---log---")
    Object ret = joinPoint.proceed();
    return ret;
  }
}
复制代码
  1. 配置切面:
代码语言:javascript复制
<bean id="arround" class="com.xxx.MyAspect" />
<!-- 告知spring基于注解进行切面开发 -->
<aop:aspectj-autoproxy />
复制代码

这样就完成了我们之前的那四个步骤,现在我们在从工厂中获取的对象就是代理对象,调用方法时,就会执行额外功能(注意: 要符合切入点表达式的方法)。

二、细节分析

  1. 切入点复用

切入点复用: 在切面中定义一个函数,上面加上@Pointcut注解,通过这种方式定义切入点表达式,实现了切入点的复用,相当于把切入点抽取了出来,方便切入点增加多个额外功能冗余的问题。这样我们就可以灵活的将切入点和额外功能进行自由组合。

代码语言:javascript复制
@Aspect // 指定是切入类
public class MyAspect{
  
  @Pointcut("execution(* login(..))")
  public void myPointCut(){}
	
  @Arround(value="myPointcut()"))
	public Object arround(ProceedingJoinPoint joinPoint) throws Throwable{
      System.out.println("---log---")
      Object ret = joinPoint.proceed();
      return ret;
  }
  
  @Arround(value="myPointcut()")
	public Object arround1(ProceedingJoinPoint joinPoint) throws Throwable{
      System.out.println("---tx---")
      Object ret = joinPoint.proceed();
      return ret;
  }
}
复制代码
  1. 动态代理的创建方式
代码语言:javascript复制
我们前面说到了Spring底层动态代理的两种方式: 
1. JDK动态代理:通过实现接口方式,创建代理对象
2. cglib动态代理: 通过继承父类的方式创建代理对象

那么我们上述代码所创建的代理对象是通过哪种方式创建的呢?
**默认情况下,AOP编程底层应用jdk的动态代理方式**

如果我们想要指定cglib进行动态代理创建,可以做如下设置
<aop:aspectj-autoproxy proxy-target-class=true />

复制代码

设置后我们可以通过断点的方式观察:

那么我们之前没用注解的时候,如何设置使用cglib动态代理呢:

代码语言:javascript复制
<aop:confg proxy-target-class="true">
	<aop:pointcut id="pc" expression="@annocation(com.xxx.Log)" />
  <aop:advisor advice-ref="around" pointcut-ref="pc" />
</aop:confg>
复制代码

三、AOP开发中的一个坑

我们在使用代理开发的过程中,有时候会遇到一个问题,就是额外功能失效的问题。我们先来看下这个问题是怎么出现的。

  1. 首先设置一个切面,增加额外功能。
代码语言:javascript复制
@Aspect // 指定是切入类
public class MyAspect{
  
  @Pointcut("execution(* *..UserServiceImpl.*(..))")
  public void myPointCut(){}
	
  @Arround(value="myPointcut()"))
	public Object arround(ProceedingJoinPoint joinPoint) throws Throwable{
      System.out.println("---log---")
      Object ret = joinPoint.proceed();
      return ret;
  }
  
  @Arround(value="myPointcut()")
	public Object arround1(ProceedingJoinPoint joinPoint) throws Throwable{
      System.out.println("---tx---")
      Object ret = joinPoint.proceed();
      return ret;
  }
}
复制代码
  1. 我们在目标方法中做调用:
代码语言:javascript复制
public class UserServiceImpl implements UserService {
  
  @Override
  public void register(User user){
    System.out.println("registe----")
      // 调用的是原始对象的login方法,---核心功能,切面功能不执行
      // 设计目的是: 调用代理对象的login方法
      this.login("abc", "123456");
 
  }
  
  @Override
  public boolean login(String name, String password){
    System.out.println("login-----")
  }
}
复制代码

此时要注意,当我们通过工厂获取UserService对象,并调用register方法的时候, register方法在执行的时候,是有额外功能执行的,但是由于register方法中又使用this调用了login,这个时候login方法是不会执行额外功能的。原因就是login方法是用this调用的,获得的并不是代理对象所以不会执行对应的额外功能。相当于是直接调用了原始类中的方法。如果此时让login方法也带上额外功能该怎么办呢,就是我们要通过代理对象去调用login方法。

这个时候就可以用我们之前文章中讲到的ApplicationContextAware来实现,把工厂注入进来,通过工厂去获取对象调用就可以了。

代码语言:javascript复制
public class UserServiceImpl implements UserService implements ApplicationContextAware{
  
  private ApplicationContext ctx;
  
  public void setApplicationContext(AppliactionContext application){
    	this.ctx = application;
  }
  
  @Override
  public void register(User user){
    System.out.println("registe----")
      // 调用的是原始对象的login方法,---核心功能,切面功能不执行
      // 设计目的是: 调用代理对象的login方法
      this.login("abc", "123456");
    	//获取代理对象
    	UserService userService = (UserService)ctx.getBean("userService");
    	userService.login("abc", "123456");
  }
  
  @Override
  public boolean login(String name, String password){
    System.out.println("login-----")
  }
}
复制代码

总结一下: 在同一个业务类中,进行业务方法间的相互调用,只有最外层方法才是加入额外功能的,内部方法通过普通方式调用,都是调用原始方法,如果想让内层的方法也调用代理对象的方法,就要通过ApplicationContextAware接口,获取现有工厂 ,进而获得代理对象

AOP总结:

0 人点赞