服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 给冰冰看的SpringAOP面试题

给冰冰看的SpringAOP面试题

2021-09-09 23:27三太子敖丙 Java教程

Spring之前已经跟学弟具体聊很详细的IOC已经循环依赖问题,接下来要接着为跟学妹们聊另外的一个模块了,那就是AOP,这也是面试官比较喜欢问的一个模块点。

给冰冰看的SpringAOP面试题

Spring之前已经跟学弟具体聊很详细的IOC已经循环依赖问题,接下来要接着为跟学妹们聊另外的一个模块了,那就是AOP,这也是面试官比较喜欢问的一个模块点。

什么是AOP?

  • AOP通常叫面向切面编程(Aspect-oriented Programming,简称AOP),它是一种编程范式,通过预编译的方式和运行期动态代理实现程序功能的统一维护的一种技术。
  • 通常用来对隔离不同业务逻辑,比如常见的事务管理、日志管理等。同时实现AOP的方式也有两种:cglib 以及 jdk两种方式来实现。

给冰冰看的SpringAOP面试题

为什么要有AOP?

假设现在有几个实现方法,需要做日志处理,正常来说我们只需要手动添加一下日志就可以了,我们都知道在真正的业务代码中,代码行数,以及方法数那是一个天文数字,如果都要手动添加那工作量不现实。

本着作为程序员因该想着怎么合理的偷懒的习惯,所以应该想办法提高效率。

AOP因此就产生了,说白了AOP就是通过某种匹配规则去匹配方法,然后再添加对应的日志处理。而AOP本身的实现方式就是通过ASM字节码框架动态生成技术,在程序运行的时候,根据需求(添加文件)动态创建字节码文件,之前讲的设计模式中动态代理模式中也有讲到,大家可以再去复习一下。

AOP的核心概念

切面(Aspect):似于 Java 中的类声明,常用于应用中配置事务或者日志管理。一般使用 @Aspect 注解或者 来定义一个切面。

连接点(Join Point):程序执行中的特定点,比如方法执行、处理一个异常等

切点(Pointcut):通过一种规则匹配的正则表达式,当有连接点可以匹配到切点时,就会触发改切点相关联的指定通知。

通知(Advice):在切面中某个连接点采取的动作,通知方式也有5种

  • around(环绕通知):前后都加
  • before(前置通知)
  • after(后置通知)
  • exception(异常通知)
  • return(返回通知)

织入(Weaving):链接切面和目标对象创建一个通知对象的过程。

AOP其实就是一种编程思想,而这上面的这个点就是编程的具体实现规范。

一个应用中可以有多种通知方式所以在AOP中引入一种设计模式责任链模式通过这这种模式来顺序执行每一个通知当然也可以使用@Order注解,配置数字越小,越先执行。关于责任链模式的大家也可以去看看我之前写的设计模式复习一下。

AOP的执行过程

之前跟大家聊IOC的时候跟大家聊过它的启动过程,同样的AOP也有指定的执行流程,但是需要IOC作为基础。

  • IOC容器启动,用来存放对象
  • 进行对象的实例化和初始化操作,将生成的完成的对象存放到容器中(容器运行中的一些对象比如BeanFactoryProcesser、methodInterceptore等还有其他的很多对象)
  • 从创建好的容器中获取需要对象
  • 调用具体的方法开始调用

说了这么多理论知识,要想知道里面的具体执行流程,还是老样子,一步一步debug进入源码查看流程了

首先还是需要先准备配置一个切面

  1. @Aspect 
  2. @Component 
  3. public class LogUtil { 
  4.  
  5.     @Pointcut("execution(public * com.ao.bing.demo.spring.aop..*.*(..))"
  6.     public void pctMethod() { 
  7.     } 
  8.  
  9.     @Around(value = "pctMethod()"
  10.     public Object around(ProceedingJoinPoint pjp) throws Throwable { 
  11.         Object ret = pjp.proceed(); 
  12.         System.out.println("Around advice"); 
  13.         return ret; 
  14.     } 
  15.  
  16.     @Before("pctMethod()"
  17.     public void before() { 
  18.         System.out.println("Before advice"); 
  19.     } 
  20.  
  21.     @After(value = "pctMethod()"
  22.     public void after() { 
  23.         System.out.println("After advice"); 
  24.     } 
  25.  
  26.     @AfterReturning(value = "pctMethod()"
  27.     public void afterReturning() { 
  28.         System.out.println("AfterReturning advice"); 
  29.     } 
  30.  
  31.     @AfterThrowing(value = "pctMethod()"
  32.     public void afterThrowing() { 
  33.         System.out.println("AfterThrowing advice"); 
  34.     } 
  35.    
  36.    
  37.     // mian方法测试demo 
  38.       public static void main(String[] args) { 
  39.         ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:applicationContext.xml"); 
  40.         AopTestDemo aopTestDemo = applicationContext.getBean(AopTestDemo.class); 
  41.         aopTestDemo.method("测试AOP"); 
  42.  
  43.     } 

这里配置一个LogUtil的切面demo,五种通知都写了一遍,同时也有一个main方法测试。准备工作做完了就正式开始debug代码了

给冰冰看的SpringAOP面试题

首先在aopTestDemo里面我们打上断点,此时我们的过去的bean对象aopTestDemo已经是通过动态代理生成的对象了,其中这里面有很多的CALLBACK方法属性。那写方法属性是什么呢?

其实这里面就又跟Spring的拦截器有关,其实就是一种设计模式观察者模式说白了就是对对象的一种行为的监听。通过回调机制来实现通知的功能。

那既然是回调方法,那就先进DynamicAdvisedInterceptor方法中

  1. private static class DynamicAdvisedInterceptor implements MethodInterceptor, Serializable { 
  2.     private final AdvisedSupport advised; 
  3.  
  4.     public DynamicAdvisedInterceptor(AdvisedSupport advised) { 
  5.         this.advised = advised; 
  6.     } 
  7.  
  8.     @Nullable 
  9.     public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { 
  10.         Object oldProxy = null
  11.         boolean setProxyContext = false
  12.         Object target = null
  13.         TargetSource targetSource = this.advised.getTargetSource(); 
  14.  
  15.         Object var16; 
  16.         try { 
  17.             if (this.advised.exposeProxy) { 
  18.                 oldProxy = AopContext.setCurrentProxy(proxy); 
  19.                 setProxyContext = true
  20.             } 
  21.  
  22.             target = targetSource.getTarget(); 
  23.             Class<?> targetClass = target != null ? target.getClass() : null
  24.           // 从advised中 获取配置好的AOP的通知方法 -重点 
  25.             List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass); 
  26.             Object retVal; 
  27.           // 如果没有配置通知方法,则直接调用target对象的调用方法 
  28.             if (chain.isEmpty() && Modifier.isPublic(method.getModifiers())) { 
  29.                 Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args); 
  30.                 retVal = methodProxy.invoke(target, argsToUse); 
  31.             } else { 
  32.            // 通过CglibMethodInvocation来启动advice通知 - 重点 
  33.                 retVal = (new CglibAopProxy.CglibMethodInvocation(proxy, target, method, args, targetClass, chain, methodProxy)).proceed(); 
  34.             } 
  35.  
  36.             retVal = CglibAopProxy.processReturnType(proxy, target, method, retVal); 
  37.             var16 = retVal; 
  38.         } finally { 
  39.             if (target != null && !targetSource.isStatic()) { 
  40.                 targetSource.releaseTarget(target); 
  41.             } 
  42.  
  43.             if (setProxyContext) { 
  44.                 AopContext.setCurrentProxy(oldProxy); 
  45.             } 
  46.  
  47.         } 
  48.  
  49.         return var16; 
  50.     } 

在DynamicAdvisedInterceptor中有一个List chain这里获取到我们配置的通知方法

给冰冰看的SpringAOP面试题

从上面的截图可以看到Spring先是把所有的通知获取到,放在一个list集合对象中,因为此时list不为空所以就又会走到 通过CglibMethodInvocation来启动advice通知这一步的流程中

  1. public class ReflectiveMethodInvocation implements ProxyMethodInvocation, Cloneable { 
  2.  
  3.  protected final Object proxy; 
  4.  
  5.  @Nullable 
  6.  protected final Object target; 
  7.  
  8.  protected final Method method; 
  9.  
  10.  protected Object[] arguments = new Object[0]; 
  11.  
  12.  @Nullable 
  13.  private final Class<?> targetClass; 
  14.  
  15.  /** 
  16.   * Lazily initialized map of user-specific attributes for this invocation. 
  17.   */ 
  18.  @Nullable 
  19.  private Map<String, Object> userAttributes; 
  20.  
  21.  /** 
  22.   * List of MethodInterceptor and InterceptorAndDynamicMethodMatcher 
  23.   * that need dynamic checks. 
  24.   */ 
  25.  protected final List<?> interceptorsAndDynamicMethodMatchers; 
  26.  
  27.  /** 
  28.   * Index from 0 of the current interceptor we're invoking. 
  29.   * -1 until we invoke: then the current interceptor. 
  30.   */ 
  31.  private int currentInterceptorIndex = -1; 
  32.  
  33.  // 省略其他的方法 
  34.  
  35.  @Override 
  36.  @Nullable 
  37.  public Object proceed() throws Throwable { 
  38.   // We start with an index of -1 and increment early. 
  39.     // 从索引为-1的拦截器还是调用,并且按顺序递增,如果整个List chain中调用完毕,则开始调用target的函数 
  40.     // 具体的实现方式在AOPUtils.invokeJoinpointUsingRefection方法中,说白了就是通过反射实现目标方法 
  41.   if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) { 
  42.    return invokeJoinpoint(); 
  43.   } 
  44.   // 获取下一个需要执行的拦截器,沿着定义好的interceptorOrInterceptionAdvice的链进行处理 
  45.   Object interceptorOrInterceptionAdvice = 
  46.     this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex); 
  47.   if (interceptorOrInterceptionAdvice instanceof InterceptorAndDynamicMethodMatcher) { 
  48.    // Evaluate dynamic method matcher here: static part will already have 
  49.    // been evaluated and found to match. 
  50.       // 对拦截器进行动态匹配,如果和定义的pointcut匹配 则就会执行当前的这个advice 
  51.    InterceptorAndDynamicMethodMatcher dm = 
  52.      (InterceptorAndDynamicMethodMatcher) interceptorOrInterceptionAdvice; 
  53.    Class<?> targetClass = (this.targetClass != null ? this.targetClass : this.method.getDeclaringClass()); 
  54.    if (dm.methodMatcher.matches(this.method, targetClass, this.arguments)) { 
  55.     return dm.interceptor.invoke(this); 
  56.    } 
  57.    else { 
  58.     // Dynamic matching failed. 
  59.     // Skip this interceptor and invoke the next in the chain. 
  60.     return proceed(); 
  61.    } 
  62.   } 
  63.   else { 
  64.    // It's an interceptor, so we just invoke it: The pointcut will have 
  65.    // been evaluated statically before this object was constructed. 
  66.       // 普通拦截器,直接调用拦截器,将当前的this(CglibMethodInvocation)作为参数保证当前实例中调用链的执行 
  67.    return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this); 
  68.   } 
  69.  } 
  70.  
  71.  // 省略其他的方法 

这里需要注意的一点 从上面的源码中currentInterceptorIndex默认值定义为-1,相当于是作为判断当前通知链式是执行完成,执行完那就是直接通过反射调用目标方法,否者向下接着执行。

  • ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this)

interceptorOrInterceptionAdvice这个方法是关键点,即具体的调用拦截器去执行具体的方法

  1. public final class ExposeInvocationInterceptor implements MethodInterceptor, PriorityOrdered, Serializable { 
  2.   
  3.  @Override 
  4.  public Object invoke(MethodInvocation mi) throws Throwable { 
  5.   MethodInvocation oldInvocation = invocation.get(); 
  6.   invocation.set(mi); 
  7.   try { 
  8.       // 具体执行 
  9.    return mi.proceed(); 
  10.   } 
  11.   finally { 
  12.    invocation.set(oldInvocation); 
  13.   } 
  14.  } 

不知道大家发现了问题没有,在刚上面截图的时候List chain中有6个通知方法,而在配置的util中只配置了5个方法,第一个这就是ExposeInvocationInterceptor这个方法,那这个到底是啥呢?

  • 其实这里是Spring引入的又一个设计模式责任链设计模式引入这个设计模式原因就是在于,配置的通知方法这么多,那spring怎么知道去执行那一个advice呢?所以通过ExposeInvocationInterceptor(暴露调用器的拦截器)作为第一个通知方法,来保证所以的通知方法按按这中连式方式执行下去。

至此SpringAOP的通知连式结构调用流程就开始了,重复开始循环调用。一直到List chain整个链全部执行完毕

当前这里面还有一些其他的逻辑需我就没有具体细说了

比如整个链有没有一种顺序执行?还是说根据代码编写的先后顺序执行?

  • org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator#sortAdvisors

在获取到整个责任链之后会经过一次排序执行,这个排序默认是使用拓扑排序方法,大家可以具体的去看看sortAdvisors这个方法

然后在执行每一种通知方法时,都会有对应的切面通知方法

  • org.springframework.aop.aspectj.AspectJAfterAdvice
  • org.springframework.aop.aspectj.AspectJAfterReturningAdvice
  • org.springframework.aop.aspectj.AspectJAfterThrowingAdvice
  • org.springframework.aop.aspectj.AspectJAroundAdvice
  • org.springframework.aop.aspectj.AspectJMethodBeforeAdvice

我们配置的五种通知对应着上面的五种处理逻辑

所以整个责任链来说通知调用链路可以理解为一张图结构:

给冰冰看的SpringAOP面试题

以上就是Spring通知整个执行的过程,总结一下

  • Spring中的五种通知,首先是通过具Spring容器的启动过程获取到具体的通知,在调用对象时,通过动态代理ASM技术,把需要执行的advice先全部放在一个chain对象的集合中,为了保证整个链的调用默认会先调用ExposeInvocationInterceptor去触发整个链式执行,在执行完每一个advice时后都会再次回到super的proceed方法中,执行下一个advice,在执行不通的advice时后有对应的切面通知方法,当所有的advice执行完毕再通过反射调用目标方法
  • 整个过程大家可以自己dubug试一下,也不是特别的麻烦!!!

AOP事务功能

AOP处理常见的配置切面处理日志等业务,还有大家也比较熟悉的那就是事务。

其实AOP的事务也是通过配置的advice的方式来执行的

给冰冰看的SpringAOP面试题

通知对advice的拆解来实现事务的功能。

  • 在看源码的时候能给我们带来很多的思考,对我们

在Spring中大家可以通过@Transactional注解来现实事务功能,看过源码的同学肯定看过TransactionAspectSupport这个类

  1. public abstract class TransactionAspectSupport implements BeanFactoryAware, InitializingBean { 
  2.  // 省略其他代码。。。。 
  3.  @Nullable 
  4.  protected Object invokeWithinTransaction(Method method, @Nullable Class<?> targetClass, 
  5.    final InvocationCallback invocation) throws Throwable { 
  6.  
  7.   // If the transaction attribute is null, the method is non-transactional. 
  8.   TransactionAttributeSource tas = getTransactionAttributeSource(); 
  9.   final TransactionAttribute txAttr = (tas != null ? tas.getTransactionAttribute(method, targetClass) : null); 
  10.   final PlatformTransactionManager tm = determineTransactionManager(txAttr); 
  11.   final String joinpointIdentification = methodIdentification(method, targetClass, txAttr); 
  12.  
  13.   if (txAttr == null || !(tm instanceof CallbackPreferringPlatformTransactionManager)) { 
  14.    // Standard transaction demarcation with getTransaction and commit/rollback calls. 
  15.    TransactionInfo txInfo = createTransactionIfNecessary(tm, txAttr, joinpointIdentification); 
  16.    Object retVal = null
  17.    try { 
  18.     // This is an around advice: Invoke the next interceptor in the chain. 
  19.     // This will normally result in a target object being invoked. 
  20.     retVal = invocation.proceedWithInvocation(); 
  21.    } 
  22.    catch (Throwable ex) { 
  23.     // target invocation exception 
  24.         // 异常回滚 重点 
  25.     completeTransactionAfterThrowing(txInfo, ex); 
  26.     throw ex; 
  27.    } 
  28.    finally { 
  29.     cleanupTransactionInfo(txInfo); 
  30.    }  
  31.       // 成功后提交 重点 
  32.    commitTransactionAfterReturning(txInfo); 
  33.    return retVal; 
  34.   } 
  35.   
  36.  // 省略其他的代码。。。。 
  37.  

因此SpringAOP的声明事务也是通过advice实现。

AOP的事务整体来说比较简单,说白了就是通过advice的从新组合来完成事务功能,当然也是Spring的强大之处,扩展性是真的高。

总结

SpringAop就跟学妹们聊完了,需要理解的就是advice的通知整个流程,但是这个过程需要大家不断的去看这个框架的源码,看源码的过程能给我们很多思考,怎么才能写出高质量代码?

为了加强理解,还是有两个比较常见的面试题

advice的通知执行流程?

  • 看完整个流程如果还是不理解我觉得可以自己debug走一遍加深自己的理解,文中我也做了总结。但是要自己真的理解才能不会被面试官问倒

AOP中的Transactional事务是怎么实现的?

  • 这个问题如果理解advice的调用流程那么也就能很简单的回答了。

原文链接:https://mp.weixin.qq.com/s/qlAvW10TRNVak1oJyHO39Q

延伸 · 阅读

精彩推荐
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7472021-02-04
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18