0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
电子发烧友
开通电子发烧友VIP会员 尊享10大特权
海量资料免费下载
精品直播免费看
优质内容免费畅学
课程9折专享价
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Spring AOP如何破解java應(yīng)用

科技綠洲 ? 來源:了不起 ? 作者:了不起 ? 2023-09-25 11:16 ? 次閱讀

前面我們看過javaassit是如何破解java應(yīng)用,核心都是AOP相關(guān)的知識(shí),今天我們看下Spring AOP是怎么回事!

Spring-AOP

spring 5.x版本

AOP面向切面編程,通過預(yù)編譯方式和運(yùn)行期間動(dòng)態(tài)代理實(shí)現(xiàn)程序功能的統(tǒng)一維護(hù)的一種技術(shù)。AOP是OOP的延續(xù),從另一視角擴(kuò)展了對(duì)面向?qū)ο缶幊痰男问?。利用AOP可以對(duì)業(yè)務(wù)邏輯的各個(gè)部分進(jìn)行隔離,從而使得業(yè)務(wù)邏輯各部分之間的耦合度降低,提高程序的可重用性,同時(shí)提高了開發(fā)的效率。

Spring AOP與IOC作為整個(gè)Spring框架最為核心的兩個(gè)部分,其意義不言而喻。在Spring中,我們都是面向Bean的裝配與管理,一個(gè)Bean是一個(gè)對(duì)象,也可以是一個(gè)代理。

概念

Aspect(切面):Aspect聲明類似于Java中的類聲明,在Aspect中會(huì)包含著一些Pointcut以及相應(yīng)的 Advice。

JointPoint(連接點(diǎn)):表示在程序中明確定義的點(diǎn),典型的包括方法調(diào)用,對(duì)類成員的訪問以及異常處理程序塊的執(zhí)行等等,它自身還可以嵌套其它joint point。

Pointcut(切點(diǎn)):按規(guī)則匹配的JointPoint,這些JointPoint或是通過邏輯關(guān)系組合起來,或是通過通配、正則表達(dá)式等方式集中起來,它定義了相應(yīng)的Advice執(zhí)行的具體地方。

Advice(通知):Advice定義了在Pointcut里面定義的程序點(diǎn)具體要做的操作,它通過before、after和around來區(qū)別是在每個(gè)JointPoint之前、之后還是代替執(zhí)行的代碼。

Target(目標(biāo)對(duì)象):將Advice織入到目標(biāo)對(duì)象.。

Weaving(織入):將Aspect和其他對(duì)象連接起來, 并創(chuàng)建增強(qiáng)(代理)對(duì)象的過程

圖片

下面從幾個(gè)方面了解Spring中如何使用AOP,你會(huì)發(fā)現(xiàn),所有的配置都是圍繞著這些概念。Spring提供了AOP代理的上下文環(huán)境,而對(duì)目標(biāo)對(duì)象的加強(qiáng)(Aspect、Advisor)都是由開發(fā)者自己完成。

Spring-aop.xml

  1. 基于aspect配置AOP,一個(gè)Aspect包含Pointcut與Advice:
< beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
  http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd" >
    < !-- 業(yè)務(wù)類 -- >
    < bean id="queryService" class="com.sucl.blog.springaop.service.QueryService"/ >
    < !-- 定義Aspect,aop:method 的來源  -- >
    < bean id="logAspect" class="com.sucl.blog.springaop.aspect.LogAspect"/ >
    
    < !-- 基于Aspect-- >
    < aop:config >
        < aop:aspect ref="logAspect" >
            < aop:pointcut id="pointcut" expression="execution(* com.sucl.blog.springaop.service..*(..))"/ >

            < aop:before method="before" pointcut-ref="pointcut"/ >
            < aop:after-returning method="afterReturning" pointcut-ref="pointcut" returning="result"/ >
            < aop:after-throwing method="afterThrowing" pointcut-ref="pointcut" throwing="e"/ >
            < aop:after method="after" pointcut-ref="pointcut"/ >
            < aop:around method="around" pointcut-ref="pointcut"/ >
        < /aop:aspect >
    < /aop:config >
< /beans >
  1. 基于advice配置AOP,這種方式更方便Advice的復(fù)用
< beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
  http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd" >

    < bean id="queryService" class="com.sucl.blog.springaop.service.QueryService"/ >
    < bean id="logAdvice" class="com.sucl.blog.springaop.advice.LogAdvice"/ >

    < !-- 基于Advisor -- >
    < aop:config >
        < aop:pointcut id="pointcut" expression="execution(* com.sucl.blog.springaop.service..*(..))"/ >
        < aop:advisor advice-ref="logAdvice" pointcut-ref="pointcut"/ >
    < /aop:config >
< /beans >

@Aspect注解

基于注解@Aspect定義切面來增強(qiáng)目標(biāo)對(duì)象,與上面的XML第一種配置對(duì)應(yīng),記得使用@EnableAspectJAutoProxy注解開啟

@Slf4j
@Aspect
public class LogAspect {

    @Pointcut("execution(* com.sucl.blog.springaop.service..*(..))")
    public void pointcut(){}

    /**
     * 方法執(zhí)行前執(zhí)行
     * @param joinPoint
     * @param arg1
     */
    @Before(value = "pointcut()")
    public void before(JoinPoint joinPoint){
        log.info(" >> > 執(zhí)行 before");
    }

    /**
     * 方法出現(xiàn)異常時(shí)執(zhí)行
     * @param joinPoint
     */
    @AfterThrowing(value = "pointcut()", throwing = "e")
    public void afterThrowing(JoinPoint joinPoint, Exception e){
        log.info(" >> > 執(zhí)行 afterThrowing: {}", e.getMessage());
    }

    /**
     * 方法返回后執(zhí)行,異常時(shí)不執(zhí)行
     * @param joinPoint
     */
    @AfterReturning(value = "pointcut()", returning = "result")
    public void afterReturning(JoinPoint joinPoint, Object result){
        log.info(" >> > 執(zhí)行 afterReturning: {}" ,result);
    }

    /**
     * 方法執(zhí)行完執(zhí)行,不管是否發(fā)生異常
     * @param joinPoint
     */
    @After(value = "pointcut()")
    public void after(JoinPoint joinPoint){
        log.info(" >> > 執(zhí)行 after");
    }

    /**
     * 在after與 around 前后執(zhí)行
     * @param pjp
     * @param jp
     */
    @Around(value = "pointcut()")
    public void around(ProceedingJoinPoint pjp){
        try {
            log.info(" >> > 執(zhí)行 around starting");
            pjp.proceed();
            log.info(" >> > 執(zhí)行 around finished");
        } catch (Throwable e) {
            log.info(" >> > 執(zhí)行 around 異常:{}", e.getMessage());
        }
    }
}

AOP代理執(zhí)行順序

spring 5.2.7之后的執(zhí)行順序:

圖片

Pointcut表達(dá)式(AspectJ)

execution:用于匹配方法執(zhí)行的連接點(diǎn)

within:用于匹配指定類型內(nèi)的方法執(zhí)行

this:用于匹配當(dāng)前AOP代理對(duì)象類型的執(zhí)行方法;注意是AOP代理對(duì)象的類型匹配,這樣就可能包括引入接口也類型匹配

target:用于匹配當(dāng)前目標(biāo)對(duì)象類型的執(zhí)行方法;注意是目標(biāo)對(duì)象的類型匹配,這樣就不包括引入接口也類型匹配

args:用于匹配當(dāng)前執(zhí)行的方法傳入的參數(shù)為指定類型的執(zhí)行方法

@within:用于匹配所以持有指定注解類型內(nèi)的方法

@target:用于匹配當(dāng)前目標(biāo)對(duì)象類型的執(zhí)行方法,其中目標(biāo)對(duì)象持有指定的注解

@args:用于匹配當(dāng)前執(zhí)行的方法傳入的參數(shù)持有指定注解的執(zhí)行

@annotation:用于匹配當(dāng)前執(zhí)行方法持有指定注解的方法

bean:Spring AOP擴(kuò)展的,AspectJ沒有對(duì)于指示符,用于匹配特定名稱的Bean對(duì)象的執(zhí)行方法

基于@Configuration

通過ProxyFactoryBean我們可以生成基于目標(biāo)對(duì)象的代理。通過下面幾行代碼,加上自定義的切面實(shí)現(xiàn)(MethodInterceptor、Advice等接口的實(shí)現(xiàn)),就可以實(shí)現(xiàn)上面的before、after、around等通知切面。

@Configuration
public class ProxyConfiguration {

    @Bean
    public ProxyFactoryBean printServiceProxy() {
        ProxyFactoryBean proxyFactoryBean = new ProxyFactoryBean();
        proxyFactoryBean.setTarget(new PrintService());
        proxyFactoryBean.setProxyTargetClass(true);
        addInterceptors(proxyFactoryBean);
        return proxyFactoryBean;
    }

    /**
     * 定義方法攔截器:(支持通配符)
     * org.springframework.aop.Advisor
     * org.aopalliance.intercept.Interceptor
     *
     * MethodBeforeAdvice
     * AfterReturningAdvice
     * ThrowsAdvice
     *
     * org.aopalliance.intercept.MethodInterceptor
     * org.aopalliance.aop.Advice
     *
     * @param proxyFactoryBean
     */
    private void addInterceptors(ProxyFactoryBean proxyFactoryBean) {
        proxyFactoryBean.setInterceptorNames("logAdvice");
    }

    @Bean
    public LogAdvice logAdvice() {
        return new LogAdvice();
    }
}

@Slf4j
public class LogAdvice implements MethodInterceptor {

    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable {
        try {
            log.info(" >> > before");
            Object result = invocation.proceed();
            log.info(" >> > afterReturning : {}", result);
            return result;
        } catch (Throwable e) {
            log.info(" >> > afterThrowing : {}", e.getMessage());
            throw e;
        } finally {
            log.info(" >> > after");
        }
    }
}

實(shí)現(xiàn)原理

這里說到的是上面第二種通過@Aspect的形式實(shí)現(xiàn)AOP功能,這種模式更適合業(yè)務(wù)模塊中對(duì)特定模塊內(nèi)的方法進(jìn)行業(yè)務(wù)代理。我們需要依賴注解 EnableAspectJAutoProxy , 下面來看看具體如何實(shí)現(xiàn)?

  1. 在EnableAspectJAutoProxy中可以看到@Import(AspectJAutoProxyRegistrar.class),如果你之前看過之前的spring boot start你就知道,這里是基于ImportBeanDefinitionRegistrar在Spring容器中注冊(cè)BeanDefinition。
  2. 可以看到AspectJAutoProxyRegistrar#registerBeanDefinitions第一行,AopConfigUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(registry), 主要是注冊(cè)了BeanPostProcessor(AnnotationAwareAspectJAutoProxyCreator)。
  3. 我們知道在Spring Bean生命中期中,Bean執(zhí)行初始化前后會(huì)執(zhí)行容器中的nPostProcessor,Spring AOP即通過AnnotationAwareAspectJAutoProxyCreator這個(gè)BeanPostProcessor完成Bean的代理工作。
  4. 其父類AbstractAutoProxyCreator中,postProcessAfterInitialization -> wrapIfNecessary -> createProxy -> ProxyFactory -> getProxy 到這里,Bean的代理對(duì)象也就生成了, 當(dāng)然省略了各種判斷以及加工過程。

代理方式

  • Cglib和JDK Proxy
    Spring AOP主要是通過這兩個(gè)代理框架來實(shí)現(xiàn)代理的。一般情況下,基于接口代理時(shí)使用JDK動(dòng)態(tài)代理,否則使用Cglib.
    • Java動(dòng)態(tài)代理只能夠?qū)涌谶M(jìn)行代理,不能對(duì)普通的類進(jìn)行代理(因?yàn)樗猩傻拇眍惖母割悶镻roxy,Java類繼承機(jī)制不允許多重繼承)。而CGLIB能夠代理普通類
    • Java動(dòng)態(tài)代理使用Java原生的反射API進(jìn)行操作,在生成類上比較高效;CGLIB使用ASM框架直接對(duì)字節(jié)碼進(jìn)行操作,在類的執(zhí)行過程中比較高效
  • AspectJ
    AspectJ是一個(gè)功能強(qiáng)大的面相切面編程框架,是對(duì)Java面向?qū)ο蟮臄U(kuò)展,支持編譯時(shí)、編譯后、加載時(shí)為目標(biāo)對(duì)象(不僅僅是類方法)織入代理。在Spring AOP中引入了aspectjweaver.jar,僅僅使用了其注解。

下面是網(wǎng)上Spring AOP與AspectJ的比對(duì)

Spring AOPAspectJ
用純Java實(shí)現(xiàn)使用Java編程語言的擴(kuò)展實(shí)現(xiàn)
無需單獨(dú)的編譯過程除非設(shè)置了LTW,否則需要AspectJ編譯器(ajc)
僅需運(yùn)行時(shí)編織運(yùn)行時(shí)編織不可用。支持編譯時(shí),后編譯和加載時(shí)編織
不足–僅支持方法級(jí)編織更強(qiáng)大–可以編織字段,方法,構(gòu)造函數(shù),靜態(tài)初始值設(shè)定項(xiàng),最終類/方法等
只能在Spring容器管理的bean上實(shí)現(xiàn)可以在所有領(lǐng)域?qū)ο笊蠈?shí)施
僅支持方法執(zhí)行切入點(diǎn)支持所有切入點(diǎn)
代理是針對(duì)目標(biāo)對(duì)象創(chuàng)建的,并且方面已應(yīng)用于這些代理在應(yīng)用程序執(zhí)行之前(運(yùn)行時(shí)之前)將方面直接編織到代碼中
比AspectJ慢得多更好的性能
易于學(xué)習(xí)和應(yīng)用比Spring AOP復(fù)雜得多

結(jié)束語

Spring AOP是整個(gè)Spring框架的核心,里面涉及到的內(nèi)容不是很多,但是都比較有深度,在spring諸多模塊中,比如事務(wù)、緩存、鑒權(quán)等等方面都有使用到。由于springboot的出現(xiàn),xml的配置形式使用 得比較少了,但是這種配置的形式更直白地體現(xiàn)了AOP需要的配置以及各個(gè)組件的依賴關(guān)系。而基于@Aspect的形式代理業(yè)務(wù)模塊中的方法更簡(jiǎn)單直觀,而基于ProxyFactoryBean、ProxyFactory的方式,在編寫類似 cache的模塊會(huì)更加的靈活。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • JAVA
    +關(guān)注

    關(guān)注

    20

    文章

    2982

    瀏覽量

    106456
  • 開發(fā)
    +關(guān)注

    關(guān)注

    0

    文章

    371

    瀏覽量

    41235
  • 程序
    +關(guān)注

    關(guān)注

    117

    文章

    3816

    瀏覽量

    82111
  • spring
    +關(guān)注

    關(guān)注

    0

    文章

    340

    瀏覽量

    14762
  • AOP
    AOP
    +關(guān)注

    關(guān)注

    0

    文章

    40

    瀏覽量

    11169
收藏 0人收藏

    評(píng)論

    相關(guān)推薦

    AOP知識(shí)詳解

    今天我們繼續(xù)看看AOP相關(guān)的知識(shí),前面說到了Javassit,Spring AOP,通過該篇,讓你對(duì)AOP有更完整的認(rèn)識(shí)。 AOP 再看
    的頭像 發(fā)表于 09-25 11:14 ?1148次閱讀
    <b class='flag-5'>AOP</b>知識(shí)詳解

    java spring教程

    java spring教程理解Spring 實(shí)現(xiàn)原理掌握Spring IOC,AOP掌握Spring
    發(fā)表于 09-11 11:09

    什么是java spring

    什么是java springSpring是一個(gè)開源框架,它由Rod Johnson創(chuàng)建。它是為了解決企業(yè)應(yīng)用開發(fā)的復(fù)雜性而創(chuàng)建的。Spring使用基本的JavaBean來完成以前只可能由EJB完成
    發(fā)表于 09-11 11:16

    Spring工作原理

    本文介紹Spring工作原理,以及IoC(Inversion of control): 控制反轉(zhuǎn)和AOP(Aspect-Oriented Programming): 面向方面編程
    發(fā)表于 07-10 07:41

    Spring筆記分享

    Spring實(shí)現(xiàn)了使用簡(jiǎn)單的組件配置組合成一個(gè)復(fù)雜的應(yīng)用。在 Spring 中可以使用XML和Java注解組合這些對(duì)象。6) 一站式:在IOC和AOP的基礎(chǔ)上可以整合各種企業(yè)應(yīng)用的開
    發(fā)表于 11-04 07:51

    Spring認(rèn)證」Spring Hello World 項(xiàng)目示例

    顯示 Java Build Path 窗口,如下所示 -現(xiàn)在使用“庫(kù)”選項(xiàng)卡下可用的“添加外部 JAR”按鈕從 Spring Framework 和 Common Logging 安裝目錄中添加以下核心
    發(fā)表于 08-17 13:49

    java的動(dòng)態(tài)代理機(jī)制和作用

    學(xué)習(xí)Spring的時(shí)候,我們知道Spring主要有兩大思想,一個(gè)是IoC,另一個(gè)就是AOP,對(duì)于IoC,依賴注入就不用多說了,而對(duì)于Spring的核心
    發(fā)表于 09-27 14:37 ?0次下載

    java動(dòng)態(tài)代理機(jī)制詳解的類和接口描述

    在學(xué)習(xí)Spring的時(shí)候,我們知道Spring主要有兩大思想,一個(gè)是IoC,另一個(gè)就是AOP,對(duì)于IoC,依賴注入就不用多說了,而對(duì)于Spring的核心
    發(fā)表于 09-28 13:33 ?0次下載

    Spring認(rèn)證_什么是Spring GraphQL

    Spring GraphQL 為構(gòu)建在 GraphQL Java 上的 Spring 應(yīng)用程序提供支持。兩個(gè)團(tuán)隊(duì)之間的聯(lián)合聯(lián)合。我們的共同理念是少固執(zhí)己見,更專注于全面和廣泛的支持。 Spri
    的頭像 發(fā)表于 08-06 14:30 ?868次閱讀
    <b class='flag-5'>Spring</b>認(rèn)證_什么是<b class='flag-5'>Spring</b> GraphQL

    Spring認(rèn)證」Spring IoC 容器

    ,我們將在下一章中討論。 容器通過讀取提供的配置元數(shù)據(jù)來獲取有關(guān)要實(shí)例化、配置和配置哪些對(duì)象的指令。數(shù)據(jù)可以由XML、Java注釋或Java代碼表示。下圖展示了Spring如何工作的高級(jí)視圖。 IoC
    的頭像 發(fā)表于 06-28 13:27 ?866次閱讀
    「<b class='flag-5'>Spring</b>認(rèn)證」<b class='flag-5'>Spring</b> IoC 容器

    Spring認(rèn)證是什么?

    ,例如:配置、組件掃描、AOP、數(shù)據(jù)訪問和事務(wù)、REST、安全、自動(dòng)配置、執(zhí)行器、 Spring boot測(cè)試等。
    的頭像 發(fā)表于 07-04 10:19 ?1478次閱讀
    <b class='flag-5'>Spring</b>認(rèn)證是什么?

    如何獲得Spring認(rèn)證?學(xué)習(xí)JAVA如何獲得Spring Professional認(rèn)證?

    、組件掃描、AOP、數(shù)據(jù)訪問和事務(wù)、REST、安全、自動(dòng)配置、執(zhí)行器、 Spring boot測(cè)試等。 1)參加Spring中國(guó)教育管理中心授權(quán)合作伙伴Spring培訓(xùn)課程 2)報(bào)名考
    的頭像 發(fā)表于 07-04 10:20 ?2031次閱讀
    如何獲得<b class='flag-5'>Spring</b>認(rèn)證?學(xué)習(xí)<b class='flag-5'>JAVA</b>如何獲得<b class='flag-5'>Spring</b> Professional認(rèn)證?

    解讀Spring源碼中的IOC和AOP部分

    Spring Framework 是一個(gè)非常流行的開源框架,為 Java 應(yīng)用程序提供了廣泛的支持和功能。
    的頭像 發(fā)表于 06-06 15:49 ?864次閱讀

    聊聊在使用Spring AOP時(shí)一個(gè)非常常見的概念A(yù)spectJ

    小伙伴們知道,Java 23 種設(shè)計(jì)模式中有一種模式叫做代理模式,這種代理我們可以將之稱為靜態(tài)代理,Spring AOP 我們常說是一種動(dòng)態(tài)代理,那么這兩種代理的區(qū)別在哪里呢?
    的頭像 發(fā)表于 08-30 09:40 ?689次閱讀
    聊聊在使用<b class='flag-5'>Spring</b> <b class='flag-5'>AOP</b>時(shí)一個(gè)非常常見的概念A(yù)spectJ

    AOP要怎么使用

    到,創(chuàng)建一個(gè)切面Advisor,并且將切點(diǎn)都綁定到一個(gè)自定義注解上面。 引入AOP的Starts: org .springframework.boot spring -boot-starter-aop
    的頭像 發(fā)表于 10-09 16:18 ?851次閱讀
    <b class='flag-5'>AOP</b>要怎么使用

    電子發(fā)燒友

    中國(guó)電子工程師最喜歡的網(wǎng)站

    • 2931785位工程師會(huì)員交流學(xué)習(xí)
    • 獲取您個(gè)性化的科技前沿技術(shù)信息
    • 參加活動(dòng)獲取豐厚的禮品