成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

關于 Spring AOP 的原理分析!

開發 后端
通過本文的分析,我們將了解了 Spring AOP的基本概念、實現機制、核心組件以及如何在實際項目中應用 AOP。

Spring AOP是 Spring框架中的一個重要模塊,它通過分離關注點來提高代碼的模塊化程度,AOP允許開發者在不改變業務邏輯的情況下,通過切面來增強或修改代碼的行為。本文我們將深入分析 Spring AOP的原理。

一、Spring AOP概述

什么是AOP?

AOP,全程 Aspect-Oriented Programming,中文翻譯為面向切面編程,它是一種編程范式,旨在通過將橫切關注點(如日志記錄、事務管理、權限控制等)分離出來,使得這些關注點可以獨立于業務邏輯進行處理。AOP的核心概念包括:

  • 切面(Aspect):模塊化的關注點,通常橫切多個對象。
  • 連接點(Join Point):程序執行過程中的某個點,比如方法調用或異常拋出。
  • 通知(Advice):在切面的某個特定的連接點上執行的動作。
  • 切入點(Pointcut):匹配連接點的斷言。
  • 目標對象(Target Object):被通知的對象。
  • 代理(Proxy):通知目標對象后,創建的對象。
  • 織入(Weaving):將切面連接到其它應用程序類型或對象上,并創建一個通知對象。

二、Spring AOP的核心原理

1.AOP的實現機制

Spring AOP基于代理模式實現,主要通過Proxy對象來替代目標對象,并在Proxy對象的方法調用中插入切面邏輯。Spring AOP使用ProxyFactory和AdvisedSupport等類來管理和創建代理對象。代理又可以細分為:

  • JDK動態代理:基于接口的代理,目標對象必須實現一個或多個接口。
  • CGLIB代理:基于子類的代理,適用于目標對象沒有實現接口的情況。

2.AOP的核心組件

  • Advisor:包含切入點和通知的元數據。
  • Advice:定義切面在連接點上執行的操作。
  • Pointcut:定義匹配連接點的規則。
  • AopProxy:負責創建代理實例,具體實現有JdkDynamicAopProxy和CglibAopProxy。

3.AOP的執行流程

Spring AOP的執行流程是理解其工作原理的關鍵,它通過在程序運行時動態地將切面邏輯織入到目標對象中,從而實現橫切關注點的分離。下面我們來詳細地分析 Spring AOP的執行流程。

(1) 配置切面

AOP的執行流程從配置切面開始,切面可以通過 XML配置文件或基于注解的方式進行定義。配置切面時,主要涉及以下幾個元素:

  • 切面類(Aspect):包含橫切邏輯的類,通常用@Aspect注解標識。
  • 通知方法(Advice):定義在特定連接點上執行的橫切邏輯。通知類型包括@Before、@After、@Around、@AfterReturning、@AfterThrowing等。
  • 切入點表達式(Pointcut Expression):用于匹配連接點的方法執行點,通常使用AspectJ的切入點表達式語法。

(2) 創建代理對象

在Spring容器啟動時,Spring會掃描配置的切面類,并為每個目標對象創建代理對象。代理對象負責在目標方法執行前后插入切面邏輯。Spring AOP使用兩種主要的代理方式:

  • JDK動態代理:適用于目標對象實現了接口的情況,通過Java的反射機制創建代理對象。
  • CGLIB代理:適用于目標對象沒有實現接口的情況,通過生成目標類的子類來創建代理。

(3) 方法調用攔截

當客戶端代碼調用目標對象的方法時,實際上是通過代理對象來進行調用的。代理對象實現了與目標對象相同的接口,因此客戶端代碼無需感知代理的存在。

  • 攔截方法調用:代理對象攔截對目標方法的調用。對于JDK動態代理,這是通過實現InvocationHandler接口的invoke方法來實現的;對于CGLIB代理,這是通過生成子類并重寫方法來實現的。

(4) 執行通知

在方法調用被攔截后,代理對象會根據切面配置執行相應的通知邏輯:

  • Before通知:在目標方法執行之前執行。
  • After通知:在目標方法執行之后執行,無論方法是否拋出異常。
  • Around通知:包圍目標方法的執行,可以在方法執行前后進行自定義邏輯,甚至可以決定是否執行目標方法。
  • AfterReturning通知:在目標方法成功返回后執行。
  • AfterThrowing通知:在目標方法拋出異常后執行。

(5) 執行目標方法

在執行完Before或Around通知的前置邏輯后,代理對象會調用目標對象的實際方法。目標方法執行完成后,代理對象會繼續執行After、Around的后置邏輯、AfterReturning或AfterThrowing通知。

(6) 返回結果或拋出異常

代理對象在完成所有通知邏輯后,將目標方法的返回結果返回給調用方。如果目標方法拋出異常,代理對象也會處理異常并根據配置決定是否重新拋出或轉換異常。

(7) 結束

AOP的執行流程在代理對象返回結果或拋出異常后結束,整個過程是透明的,調用方無需關心代理的存在,目標對象的行為在運行時被增強。

三、Spring AOP核心源碼分析

Spring AOP的源碼涉及到多個核心類和接口,包括ProxyFactory、AdvisedSupport、AopProxy、JdkDynamicAopProxy、CglibAopProxy等。下面,我們將對這些核心組件進行詳細分析。

1.ProxyFactory

ProxyFactory是Spring AOP創建代理的核心工廠類。它負責根據配置創建合適的代理對象(JDK動態代理或CGLIB代理)。

public class ProxyFactory extends ProxyCreatorSupport {

    // 獲取代理對象
    public Object getProxy() {
        return createAopProxy().getProxy();
    }

    // 創建AopProxy對象
    protected AopProxy createAopProxy() {
        if (!this.isProxyTargetClass()) { // 是否強制使用CGLIB代理
            return new JdkDynamicAopProxy(this);
        }
        return new CglibAopProxy(this);
    }
}
  • getProxy():對外提供獲取代理對象的方法。
  • createAopProxy():根據ProxyTargetClass屬性判斷使用JDK動態代理還是CGLIB代理。

2.AdvisedSupport

AdvisedSupport是Spring AOP的配置類,持有AOP代理需要的各種配置,包括目標對象、切面、通知等。

public class AdvisedSupport extends ProxyConfig implements Advised {

    private TargetSource targetSource;
    private List<Advisor> advisors = new ArrayList<>();
    private List<Class<?>> interfaces = new ArrayList<>();

    // 其他配置和方法
}
  • TargetSource:封裝了目標對象。
  • advisors:存儲應用于目標對象的通知(Advice)和切入點(Pointcut)。
  • interfaces:代理對象需要實現的接口列表。

3.AopProxy接口

AopProxy是一個接口,定義了AOP代理對象的創建方法。

public interface AopProxy {
    Object getProxy();
    Object getProxy(ClassLoader classLoader);
}
  • getProxy():用于創建代理對象。
  • getProxy(ClassLoader classLoader):允許指定類加載器創建代理對象。

4.JdkDynamicAopProxy

JdkDynamicAopProxy實現了AopProxy接口,使用JDK動態代理為目標對象創建代理。

public class JdkDynamicAopProxy implements AopProxy, InvocationHandler {

    private final AdvisedSupport advised;

    public JdkDynamicAopProxy(AdvisedSupport config) {
        this.advised = config;
    }

    @Override
    public Object getProxy() {
        return getProxy(Thread.currentThread().getContextClassLoader());
    }

    @Override
    public Object getProxy(ClassLoader classLoader) {
        return Proxy.newProxyInstance(classLoader, this.advised.getProxiedInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, this.advised.getTargetClass());

        if (chain.isEmpty()) {
            return method.invoke(this.advised.getTargetSource().getTarget(), args);
        }

        MethodInvocation invocation = new ReflectiveMethodInvocation(proxy, this.advised.getTargetSource().getTarget(), method, args, chain);
        return invocation.proceed();
    }
}
  • getProxy():通過Proxy.newProxyInstance創建代理對象。
  • invoke():實現InvocationHandler接口的方法,負責方法調用的攔截和通知鏈的執行。

5.CglibAopProxy

CglibAopProxy同樣實現了AopProxy接口,使用CGLIB庫為目標對象創建代理。

public class CglibAopProxy implements AopProxy {

    private final AdvisedSupport advised;

    public CglibAopProxy(AdvisedSupport config) {
        this.advised = config;
    }

    @Override
    public Object getProxy() {
        return getProxy(Thread.currentThread().getContextClassLoader());
    }

    @Override
    public Object getProxy(ClassLoader classLoader) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(this.advised.getTargetClass());
        enhancer.setInterfaces(this.advised.getProxiedInterfaces());
        enhancer.setCallback(new DynamicAdvisedInterceptor(this.advised));
        return enhancer.create();
    }

    private static class DynamicAdvisedInterceptor implements MethodInterceptor {

        private final AdvisedSupport advised;

        public DynamicAdvisedInterceptor(AdvisedSupport advised) {
            this.advised = advised;
        }

        @Override
        public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
            List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, this.advised.getTargetClass());

            if (chain.isEmpty()) {
                return proxy.invokeSuper(obj, args);
            }

            MethodInvocation invocation = new CglibMethodInvocation(obj, this.advised.getTargetSource().getTarget(), method, args, proxy, chain);
            return invocation.proceed();
        }
    }
}
  • getProxy():使用CGLIB的Enhancer創建代理對象。
  • DynamicAdvisedInterceptor:CGLIB的攔截器實現,負責方法調用的攔截和通知鏈的執行。

6.MethodInvocation

MethodInvocation接口及其實現類(如ReflectiveMethodInvocation)負責封裝方法調用的上下文信息,并管理通知鏈的執行。

public interface MethodInvocation extends Joinpoint {
    Method getMethod();
    Object[] getArguments();
}

public class ReflectiveMethodInvocation implements MethodInvocation {

    private final Object proxy;
    private final Object target;
    private final Method method;
    private final Object[] arguments;
    private final List<?> interceptorsAndDynamicMethodMatchers;
    private int currentInterceptorIndex = -1;

    @Override
    public Object proceed() throws Throwable {
        if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
            return this.method.invoke(this.target, this.arguments);
        }

        Object interceptorOrInterceptionAdvice = this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex);
        if (interceptorOrInterceptionAdvice instanceof MethodInterceptor) {
            MethodInterceptor interceptor = (MethodInterceptor) interceptorOrInterceptionAdvice;
            return interceptor.invoke(this);
        } else {
            return proceed();
        }
    }
}
  • proceed():遞歸調用通知鏈中的下一個攔截器,最終執行目標方法。

通過對 Spring AOP源碼的詳細分析,我們可以看到Spring AOP是如何通過代理模式實現面向切面編程的。

四、Spring AOP應用示例

下面我們通過一個簡單的 Spring AOP示例,展示如何通過AOP實現日志記錄。

1.定義業務類

public class UserService {
    public void createUser(String username) {
        System.out.println("Creating user: " + username);
    }
}

2.定義切面

@Aspect
public class LoggingAspect {

    @Before("execution(* UserService.createUser(..))")
    public void logBefore(JoinPoint joinPoint) {
        System.out.println("Before method: " + joinPoint.getSignature().getName());
    }
}

3.Spring配置

使用Java配置:

@Configuration
@EnableAspectJAutoProxy
public class AppConfig {

    @Bean
    public UserService userService() {
        return new UserService();
    }

    @Bean
    public LoggingAspect loggingAspect() {
        return new LoggingAspect();
    }
}

4.測試AOP功能

public class AopTest {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
        UserService userService = context.getBean(UserService.class);
        userService.createUser("Alice");
    }
}

輸出結果:

Before method: createUser
Creating user: Alice

五、總結

Spring AOP通過代理模式實現了面向切面編程,能夠在不改變業務邏輯的情況下增強代碼功能。通過本文的分析,我們了解了 Spring AOP的基本概念、實現機制、核心組件以及如何在實際項目中應用 AOP。Spring AOP的強大之處在于其靈活性和可擴展性,使得開發者可以輕松地實現橫切關注點的分離和復用。

責任編輯:趙寧寧 來源: 猿java
相關推薦

2011-09-15 10:15:30

Spring

2021-05-06 18:17:52

SpringAOP理解

2012-09-28 10:20:14

IBMdw

2012-09-27 09:47:43

SpringJava面向對象

2018-10-25 16:20:23

JavaSpring AOPSpringMVC

2022-05-26 09:03:39

AOP編程

2025-01-07 09:16:16

2009-06-19 13:28:30

Spring AOPSpring 2.0

2022-06-07 07:58:45

SpringSpring AOP

2009-06-22 10:41:34

Spring.AOP

2022-02-17 13:39:09

AOP接口方式

2023-03-31 08:22:48

javassitcglibAOP

2009-06-18 14:54:52

Spring AOP

2019-11-29 16:21:22

Spring框架集成

2022-06-08 08:04:28

Springservicerepository

2009-06-19 11:09:27

Spring AOP

2022-07-01 09:39:58

SpringAOPIOC

2009-09-29 10:00:40

Spring AOP框

2011-04-26 09:33:04

SpringAOP

2010-01-22 17:56:31

千兆路由交換機
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲国产精品日本 | 国产精品一区视频 | www.日韩系列 | 九九福利 | 久久亚洲国产 | 久久99精品久久久 | 最新国产精品视频 | 欧美国产视频 | 亚洲久久在线 | 亚洲午夜av | 日韩美女一区二区三区在线观看 | 伊人狠狠| 欧美精品99 | 欧美自拍第一页 | 黄色网址av | 亚洲精品一区二区三区丝袜 | 久久国产成人精品国产成人亚洲 | 不卡一区二区三区四区 | 久久久成人网 | 亚洲精品日韩在线 | 亚洲成人自拍 | 精品国产免费一区二区三区演员表 | 日本亚洲欧美 | 免费一级欧美在线观看视频 | 91中文视频 | 最新国产精品 | 亚洲视频在线一区 | 久久99精品久久久 | 狠狠干天天干 | 欧美精品一区二区免费 | 91国产视频在线观看 | 日韩电影中文字幕 | 337p日本欧洲亚洲大胆 | 在线a视频 | 久久电影一区 | 七七婷婷婷婷精品国产 | 国产99视频精品免视看9 | 香蕉婷婷| 久久久久国产成人精品亚洲午夜 | www日本高清 | 亚洲精品二区 |