前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Spring5系列(十一) | 基于注解的AOP编程

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

作者头像
一缕82年的清风
修改2021-12-13 09:44:09
2270
修改2021-12-13 09:44:09
举报
文章被收录于专栏:lsqingfeng

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

我们前面讲解了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 />

复制代码

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

image.png
image.png

那么我们之前没用注解的时候,如何设置使用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总结:

image.png
image.png
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021/04/20 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、 开发步骤
  • 二、细节分析
  • 三、AOP开发中的一个坑
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档