2019独角兽企业重金招聘Python工程师标准>>>

关于AOP的个人理解

AOP联盟定义的AOP体系结构把与AOP相关的概念大致分为了由高到低、从使用到实现的三个层次。关于这个体系结构,个人的理解是这样的,从上往下,最高层是语言和开发环境,在这个环境中可以看到几个重要的概念:base可以视为待增强对象,或者说目标对象;aspect指切面,通常包含对于base的增强应用;configuration可以看成是一种编织或者说配置,通过在AOP体系中提供这个configuration配置环境,可以把base和aspect结合起来,从而完成切面对目标对象的编织实现。 对Spring平台或者说生态系统来说,AOP是Spring框架的核心功能模块之一。AOP与IOC容器的结合使用, 为应用开发或者Spring自身功能的扩展都提供了许多便利。Spring AOP的实现和其他特性的实现一样,非常丰富,除了可以使用Spring本身提供的AOP实现之外,还封装了业界优秀的AOP解决方案AspectJ来让应用使用。在这里,主要对Spring自身的AOP实现原理做一些解析;在这个AOP实现中,Spring充分利用了IOC容器Proxy代理对象以及AOP拦截器的功能特性,通过这些对AOP基本功能的封装机制,为用户提供了AOP的实现框架。所以,要了解这些AOP的基本实现,需要我们对Java 的Proxy机制有一些基本了解。

AOP实现的基本线索

AOP实现中,可以看到三个主要的步骤,一个是代理对象的生成,然后是拦截器的作用,然后是Aspect编织的实现。AOP框架的丰富,很大程度体现在这三个具体实现中,所具有的丰富的技术选择,以及如何实现与IOC容器的无缝结合。毕竟这也是一个非常核心的模块,需要满足不同的应用需求带来的解决方案需求。 在Spring AOP的实现原理中,我们主要举ProxyFactoryBean的实现作为例子和实现的基本线索进行分析;很大一个原因,是因为ProxyFactoryBean是在Spring IoC环境中,创建AOP应用的最底层方法,从中,可以看到一条实现AOP的基本线索。在ProxyFactoryBean中,它的AOP实现需要依赖JDK或者CGLIB提供的Proxy特性。从FactoryBean中获取对象,是从getObject()方法作为入口完成的。然后为proxy代理对象配置advisor链,这个配置是在initializeAdvisorChain方法中完成的;然后就为生成AOP代理对象做好了准备,生成代理对象如下所示:

private synchronized Object getSingletonInstance() {
    if (this.singletonInstance == null) {
        this.targetSource = freshTargetSource();
        if (this.autodetectInterfaces && getProxiedInterfaces().length == 0 && !isProxyTargetClass()) {
            // Rely on AOP infrastructure to tell us what interfaces to proxy.  
            Class targetClass = getTargetClass();
            if (targetClass == null) {
                throw new FactoryBeanNotInitializedException("Cannot determine target class for proxy");
            }
            // 这里设置代理对象的接口     
            // setInterfaces(ClassUtils.getAllInterfacesForClass(targetClass, this.proxyClassLoader));  
        }
        // Initialize the shared singleton instance.  
        super.setFrozen(this.freezeProxy);
        // 注意这里的方法会使用ProxyFactory来生成我们需要的Proxy  
        this.singletonInstance = getProxy(createAopProxy());
    }
    return this.singletonInstance;
}
 
//使用createAopProxy返回的AopProxy来得到代理对象  
protected Object getProxy(AopProxy aopProxy) {
    return aopProxy.getProxy(this.proxyClassLoader);
}

上面我们看到了在Spring中通过ProxyFactoryBean实现AOP功能的第一步,得到AopProxy代理对象的基本过程,下面我们看看AopProxy代理对象的拦截机制是怎样发挥作用,是怎样实现AOP功能的。我们知道,对代理对象的生成,有CGLIB和JDK两种生成方式,在CGLIB中,对拦截器设计是通过在Cglib2AopProxy的AopProxy代理对象生成的时候,在回调DynamicAdvisedInterceptor对象中实现的,这个回调的实现在intercept方法中完成。对于AOP是怎样完成对目标对象的增强的,这些实现是封装在AOP拦截器链中,由一个个具体的拦截器来完成的。具体拦截器的运行是在以下的代码实现中完成的,这些调用在ReflectiveMethodInvocation中。

public Object proceed() throws Throwable {
    //  We start with an index of -1 and increment early.  
    //如果拦截器链中的拦截器迭代调用完毕,这里开始调用target的函数,这个函数是通过反射机制完成的,具体实现在:AopUtils.invokeJoinpointUsingReflection方法里面。  
    if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
        return invokeJoinpoint();
    }
    //这里沿着定义好的 interceptorOrInterceptionAdvice链进行处理。  
    Object interceptorOrInterceptionAdvice =
            this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex);
    if (interceptorOrInterceptionAdvice instanceof InterceptorAndDynamicMethodMatcher) {
        // Evaluate dynamic method matcher here: static part will already have  
        // been evaluated and found to match.  
        //这里对拦截器进行动态匹配的的判断,还记得我们前面分析的pointcut吗?这里是触发进行匹配的地方,如果和定义的pointcut匹配,那么这个advice将会得到执行。  
        InterceptorAndDynamicMethodMatcher dm =
                (InterceptorAndDynamicMethodMatcher) interceptorOrInterceptionAdvice;
        if (dm.methodMatcher.matches(this.method, this.targetClass, this.arguments)) {
            return dm.interceptor.invoke(this);
        } else {
            // Dynamic matching failed.  
            // Skip this interceptor and invoke the next in the chain.  
            // //如果不匹配,那么这个proceed会被递归调用,直到所有的拦截器都被运行过为止。  
            return proceed();
        }
    } else {
        // It's an interceptor, so we just invoke it: The pointcut will have  
        // been evaluated statically before this object was constructed.  
        //如果是一个interceptor,直接调用这个interceptor对应的方法  
        return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this);
    }
}

在调用拦截器的时候,我们接下去就可以看到对advice的通知的调用。而经过一系列的注册,适配的过程以后,拦截器在拦截的时候,会调用到预置好的一个通知适配器,设置通知拦截器,这是一系列Spring设计好为通知服务的类的一个,是最终完成通知拦截和实现的地方,非常的关键。比如,对MethodBeforeAdviceInterceptor的实现是这样的:

public class MethodBeforeAdviceInterceptor implements MethodInterceptor, Serializable {
    private MethodBeforeAdvice advice;
 
    /**
     * Create a new MethodBeforeAdviceInterceptor for the given advice.
     *
     * @param advice the MethodBeforeAdvice to wrap
    */
    public MethodBeforeAdviceInterceptor(MethodBeforeAdvice advice) {
        Assert.notNull(advice, "Advice must not be null");
        this.advice = advice;
    }
 
    //这个invoke方法是拦截器的回调方法,会在代理对象的方法被调用的时候触发回调。  
    public Object invoke(MethodInvocation mi) throws Throwable {
        this.advice.before(mi.getMethod(), mi.getArguments(), mi.getThis());
        return mi.proceed();
    }
}

在代码中,可以看到,就是这里,会调用advice的before方法!这样就成功的完成了before通知的编织! 因为Spring AOP本身并不打算成为一个一统天下的AOP框架,秉持Spring的一贯设计理念,设想中的Spring设计目标应该是,致力于AOP框架与IOC容器的紧密集成,通过集成AOP技术为JavaEE应用开发中遇到的普遍问题提供解决方案,从而为AOP用户使用AOP技术提供最大的便利,从这个角度上为Java EE的应用开发人员服务。在没有使用第三方AOP解决方案的时候,Spring通过虚拟机的Proxy特性和CGLIB实现了AOP的基本功能,我想,如果有了Spring AOP实现原理的知识背景,再加上我们对源代码实现的认真解读,可以为我们了解其他AOP框架与IOC容器的集成原理,也打下了很好的基础,并真正了解一个AOP框架是在怎样实现的。 这还真是就是我们喜欢开源软件一个原因,有了源代码,软件就没有什么神秘的面纱了!本立而道生,多读源代码吧,或者找一本从源代码出发讲解软件实现的书来看看,就像以前我们学习操作系统,学习TCP/IP那样!一定会有长进的。

欢迎工作一到五年的Java工程师朋友们加入Java架构开发:855801563

本群提供免费的学习指导 架构资料 以及免费的解答

不懂得问题都可以在本群提出来 之后还会有职业生涯规划以及面试指导

同时大家可以多多关注一下小编 纯干货 大家一起学习进步

转载于:https://my.oschina.net/u/3959491/blog/2186887

深入解析Spring架构与设计原理-AOP相关推荐

  1. spring技术内幕——深入解析spring架构与设计原理

    林纳斯·托瓦兹(Linus Torvalds)说:"我从心底认为,优秀的程序员与平庸的程序员之间的区别,是在于认为自己的代码重要还是数据结构更加重要.平庸的程序员眼里只有代码,优秀的程序员则 ...

  2. 《SPRING技术内幕:深入解析SPRING架构与设计原理(第2版)》

    链接:https://pan.baidu.com/s/1eM8Dgwo_lxMqJSTmQPafRw 提取码:13sb 解压密码:习习

  3. 深入剖析Spring架构与设计原理(一)

    深入剖析Spring架构与设计原理(一) 1. IOC的基础 IOC/AOP是Spring的基础, 是Spring平台实现的核心部分. 虽然,我们一开始大多只是在这个层面上,做一些配置和外部特性的使用 ...

  4. Spring Cloud Feign设计原理

    点击关注,快速进阶高级架构师 作者:亦山札记 什么是Feign? Feign 的英文表意为"假装,伪装,变形", 是一个http请求调用的轻量级框架,可以以Java接口注解的方式调 ...

  5. 深度解析LSTM神经网络的设计原理

    引人入胜的开篇: 想要搞清楚LSTM中的每个公式的每个细节为什么是这样子设计吗?想知道simple RNN是如何一步步的走向了LSTM吗?觉得LSTM的工作机制看不透?恭喜你打开了正确的文章! 前方核 ...

  6. 说说大型网站可伸缩性架构的设计原理

    可伸缩性架构指的是:不改变网站的软硬件设计,只通过改变部署的服务器数量就可以扩大或缩小网站的服务处理能力. 大型网站中的 "大型",可以表现在以下几个方面: * 用户方面 - 大量 ...

  7. 【Spring Boot实战】源码解析Spring Boot自动配置原理

    一.简介 Spring致力于让Java开发更简单,SpringBoot致力于让使用Spring进行Java开发更简单,SpringCloud致力于基于SpringBoot构建微服务生态圈,让微服务开发 ...

  8. JQuery源码解析-整体架构的设计

    最近在看jQuery的源码,查了一些资料,偶尔看到了网易云课堂里有套视频是对源码进行讲解的,是叫妙味课堂出的,看着还不错,对这方面感兴趣的推荐看下, 个人感觉很不错.所以发些文章,边学边记录. 版本: ...

  9. spring boot 异常设计原理

    spring boot 异常模板 查看浏览器请求应答信息如下: 请求返回信息以页面形式返回 : Content-Type: text/html;charset=UTF-8 根据返回页面类型查看spri ...

最新文章

  1. 服务器部署基础知识_我在生产部署期间学到的知识
  2. 一周AI看点 | 董明珠投资150亿洛阳造机器人 北京首条无人驾驶地铁线空载试运行
  3. KDD 2021 即将开幕!当线上 AI 顶会开始内卷.....
  4. html提交表单到Servlet
  5. JavaScript 开发者最期待的年度大会,惊喜不断,错过还要等好久!
  6. 参数化查询(简单举例)
  7. Java线程新特征——Java并发库
  8. HDMI光端机是什么?hdmi光端机产品参数及性能特点介绍
  9. windows安装mysql5.7
  10. Java笔记-java web实现验证码
  11. vue监听字符串长度_Vue 的 computed 和 watch 的区别
  12. 家人重病什么心情都没了
  13. 搜狐提出畅游私有化要约 后者收盘涨近50%
  14. 使用frp实现将内网映射到公网 无需花生壳
  15. SQLyog下载地址—Mysql的可视化(建议收藏)
  16. [原创]Linux系统启动过程分析
  17. Skipped xxx -- Node remains in conflict
  18. 全新荣耀手表GS 3正式发布,8通道心率AI引擎精准高效管理健康生活
  19. 【 云原生 | kubernetes 】- tekton构建CI/CD流水线(二)
  20. 【C++笔记】 判断两个数互质(做大公约数为1)

热门文章

  1. 再造一个爱因斯坦?GPT-3 让我看到了名人“重生”的希望
  2. 对网络体系变革的思考
  3. 国际运营商智慧城市探索与实践
  4. 宇宙和你,本质上其实只是个八维数字?
  5. Google提出新型学习范式「Deep Memory」,或将彻底改变机器学习领域
  6. 从空军机长到大厂程序员,关系数据库之父传奇
  7. 每日一道面试题(第7期)---Android补间动画与属性动画的区别
  8. 我们每天都在做无用功?
  9. 随笔 | 抢红包不是一件小事
  10. oracle函数,oracle编写函数