91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Spring中AOP的概念是什么

發布時間:2022-06-13 09:53:02 來源:億速云 閱讀:159 作者:zzz 欄目:開發技術

這篇文章主要介紹“Spring中AOP的概念是什么”,在日常操作中,相信很多人在Spring中AOP的概念是什么問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Spring中AOP的概念是什么”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!

Spring容器包含兩個重要的特性:面向切面編程(AOP)和控制反轉(IOC)。面向切面編程是面向對象(OOP)的一種補充,在面向對象編程的過程中編程針對的目標是一個個對象,而面向切面編程中編程針對的目標是一個個切面。切面支持跨類型跨對象(如事務的切面可以加在任何地方)進行模塊化。

前言

AOP是Spring的關鍵特性之一,雖然Spring的IOC特性并不依賴于AOP(意味著你可以只使用Spring的IOC特性而不使用AOP特性),但是二者結合起來可以靈活的實現很多中間件解決方案。比如我們經常使用的事務(@Transaction),就是通過AOP方案實現的。本文重點介紹AOP編程中的一些術語,這些術語不僅僅局限于Spring,它適用于所有的AOP編程。

Spring中AOP的概念是什么

  • 切面(Aspect):面向切面編程可以跨類跨對象進行切面編程,一個切面就是對一類橫切關注點的模塊化。

  • 切入點(JoinPoint):程序執行過程中的一個點,如方法調用、字段訪問和異常拋出等。

  • 增強(Advice):用于對切面增強,包含前增強、后增強和環繞增強。大多數AOP框架會對切入點進行攔截,并在切入點維護一個攔截器鏈。

  • 目標對象(TargetObject):包含一個或者多個切面的對象。

  • AOP代理(AOPProxy):通過Java動態代理或者CGLib增強得到的代理對象。

  • 織入(Weaving):將切面整合到完整的流執行流程。

Spring的AOP的功能和目標

Spring的AOP使用純Java語言實現(如AspectJ就不是Java語言),不需要任何額外的編譯流程,不需要修改類加載器,適用于任何Servlet容器和應用服務。Spring的AOP只支持方法攔截,不支持字段攔截,如果用戶需要使用字段攔截,可以考慮引入AspectJ等類似的框架。

Spring的AOP框架和其它的框架有些不同,Spring的Aop框架不僅僅是為了提供一個AOP功能,它更重要的功能是和Spring的IOC容器結合,提供一些企業應用服務的解決方案(如事務等),我們可以和定義一個普通Bean一樣的方式去定以一個切面。Spring的AOP不支持非常細粒度的AOP,只支持對容器中的Bean進行AOP,如果需要更細粒度的AOP,可以考慮使用AspectJ。Spring容器的一個優秀的特性就是非侵入性,所以你可以靈活的選擇自己的AOP方案,不一定非要使用Spring的方案。

代理方式

Spring對實現接口的方法默認使用Java動態代理實現AOP攔截,對于非接口方法則會使用CGLIB字節碼工具實現代理。

@AspectJ的支持

@AspectJ注解可以把一個普通的Java類聲明為切面。@AspectJ注解是AspectJ5引入的注解,Spring雖然可以讀取AspectJ5的注解用于切面元數據的配置,但是在運行的時候使用的仍然是Spring AOP進行代理,而沒有使用AspectJ的編譯器或者織入邏輯。我們會在后續討論如何在Spring中使用AspectJ的編譯器和織入邏輯。

啟用@AspectJ

Spring默認沒有啟用AspectJ,如果你需要Spring支持@AspectJ注解對應的切面,可以通過在配置類上添加注解或者使用XML啟用配置(AspectJ所在的包是:aspectjweaver.jar)。

通過Java注解啟用AspectJ注解支持:

@Configuration
@EnableAspectJAutoProxy
public class AppConfig {
}

通過XML配置啟用AspectJ注解

<aop:aspectj-autoproxy/>

定義一個切面

當啟用AspectJ支持之后,開發者定義的任何Aspect切面會自動地被檢測到,然后Spring AOP會對切面進行攔截。下面兩個例子展示了如何配置AspectJ切面,你可以通過添加@Component注解把切面Bean注冊到Spring容器中。

<bean id="myAspect" class="org.xyz.NotVeryUsefulAspect">
    <!-- configure properties of the aspect here -->
</bean>
package org.xyz;
import org.aspectj.lang.annotation.Aspect;
@Aspect
public class NotVeryUsefulAspect {

}

聲明一個切入點

切入點程序運行過程中我們感興趣的一個點,Spring的AOP框架只支持發現對Spring Bean方法上的切入點,因此你可以簡單的把切入點理解為SpringBean的方法。

切入點確定感興趣的連接點,從而使我們能夠控制何時運行通知。springaop只支持springbean的方法執行連接點,因此可以將切入點看作與springbean上方法的執行相匹配。切入點聲明由兩部分組成:一部分是由名稱和任何參數組成的簽名,另一部分是確定我們感興趣的方法執行的切入點表達式。在AOP的@AspectJ注釋樣式中,切入點簽名由常規方法定義提供,切入點表達式由@pointcut注釋指示(用作切入點簽名的方法必須具有void返回類型)。切入點由兩部分組成,一部分是用于區別不同切入點的標識(下面例子中的 private void anyOldTransfer() {} )),另外一部分是確定我們感興趣的Bean方法的表達式(下面例子中的 @Pointcut("execution(* transfer(..))") ), 下面的例子展示了一個切入點的定義:

@Pointcut("execution(* transfer(..))") // the pointcut expression
private void anyOldTransfer() {} // the pointcut signature

Spring匹配切入點的語法使用了AspectJ5中的表達式語法,我們可以參考AspectJ文檔相關的語法.

常見的切入點匹配表達

Spring支持下面常見的AspectJ切面定義語法:

  • execution:用于匹配方法的連接點。

@Pointcut("execution(public * *(..))")
private void anyPublicOperation() {}
  • within:用于匹配類型內的方法。

@Pointcut("within(com.xyz.myapp.trading..*)")
private void inTrading() {}
  • this:匹配當前AOP代理對象的執行方法

@target(org.springframework.transaction.annotation.Transactional)
  • target:target匹配目標對象的類型,即被代理對象的類型,例如A繼承了B接口,則使用target("B"),target("A")均可以匹配到A

//    當前AOP對象實現了 IPointcutService接口的任何方法
@Pointcut("target(cn.javass.spring.chapter6.service.IPointcutService)")
private void anyPublicOperation() {}
  • args:用于限定切點方法的參數類型。

args(java.io.Serializable)
  • @target:被代理對象應該包含指定的注解。

@target(org.springframework.transaction.annotation.Transactional)
  • @args: 被代理對象的參數包含指定的注解。

@args(com.xyz.security.Classified)
  • @within: 被代理的對象應包含指定注解

@within(org.springframework.transaction.annotation.Transactional)
  • @annotation:切入點包含指定的注解。

@annotation(org.springframework.transaction.annotation.Transactional)

我們可以通過“&&”和“||”對多個條件進行組合,AspectJ還有很多其它的表達式,但是Spring不支持除上述表達式以外的其它表達式。AspectJ其它表達式包含: call, get, set, preinitialization, staticinitialization, initialization, handler, adviceexecution, withincode, cflow, cflowbelow, if, @this, @withincode等。

我們在使用Spring的代理方法之前,應該知道其代理原理。Java動態代理只能攔截public接口方法上的調用,CGLib只能攔截public、protected和defult方法。如果你需要更深層次的攔截,可以考慮使用底層的Aspectj。

切面的增強

我們在上面的步驟定義好了一個切入點,我們現在就可以對這個切入點進行額外操作,這些額外操作被稱為增強,Spring支持四種增強方式:前增強、后增強、異常增強和環繞增強。Spring支持在增強方法的定義上直接定義切入點。

前增強BeforeAdvice

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class BeforeExample {
    @Before("com.xyz.myapp.CommonPointcuts.dataAccessOperation()")
    public void doAccessCheck() {
        // ...
    }
}
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class BeforeExample {

    @Before("execution(* com.xyz.myapp.dao.*.*(..))")
    public void doAccessCheck() {
        // ...
    }
}

后增強

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.AfterReturning;
@Aspect
public class AfterReturningExample {
    @AfterReturning(
        pointcut="com.xyz.myapp.CommonPointcuts.dataAccessOperation()",
        returning="retVal")
    public void doAccessCheck(Object retVal) {
        // ...
    }
}

異常增強

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.AfterThrowing;
@Aspect
public class AfterThrowingExample {
    @AfterThrowing(
        pointcut="com.xyz.myapp.CommonPointcuts.dataAccessOperation()",
        throwing="ex")
    public void doRecoveryActions(DataAccessException ex) {
        // ...
    }
}

環繞增強

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.ProceedingJoinPoint;
@Aspect
public class AroundExample {
    @Around("com.xyz.myapp.CommonPointcuts.businessService()")
    public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable {
        // start stopwatch
        Object retVal = pjp.proceed();
        // stop stopwatch
        return retVal;
    }
}

代理機制

我們前面說過,Spring AOP通過動態代理和CGLIB實現AOP對象的代理。我們可以通過如下配置設置動態代理全部走CGLIB。

<aop:config proxy-target-class="true">
    <!-- other beans defined here... -->
</aop:config>

代理工廠的使用

Spring AOP實現代理的核心類是 AspectJProxyFactory ,我們可以使用這個類編程式生成代理對象:

// create a factory that can generate a proxy for the given target object
AspectJProxyFactory factory = new AspectJProxyFactory(targetObject);

// add an aspect, the class must be an @AspectJ aspect
// you can call this as many times as you need with different aspects
factory.addAspect(SecurityManager.class);

// you can also add existing aspect instances, the type of the object supplied must be an @AspectJ aspect
factory.addAspect(usageTracker);

// now get the proxy object...
MyInterfaceType proxy = factory.getProxy()

到此,關于“Spring中AOP的概念是什么”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

大姚县| 会同县| 涞水县| 太仆寺旗| 孙吴县| 固阳县| 巴里| 怀宁县| 偏关县| 千阳县| 南通市| 平安县| 沈阳市| 民县| 和顺县| 汉川市| 汉寿县| 高青县| 凤冈县| 广安市| 司法| 普陀区| 安庆市| 西平县| 高密市| 涟源市| 上饶市| 睢宁县| 彰化市| 澄迈县| 衡东县| 鄂尔多斯市| 福贡县| 望都县| 贡嘎县| 汪清县| 油尖旺区| 拉萨市| 南汇区| 岑巩县| 上饶市|