aop源码分析(一)——开启aop到代理类生成(未完成)

简介: 我们都知道spring aop的底层是动态代理,这篇博客会分析下spring aop的实现原理。当然,spring aop的实现还是非常复杂的,这里会拆成多篇博客分析。

我们都知道spring aop的底层是动态代理,这篇博客会分析下spring aop的实现原理。当然,spring aop的实现还是非常复杂的,这里会拆成多篇博客分析。

在分析spring aop之前,先弄明白spring aop实现了个什么样的功能。

从配置文件开始

先来看个例子(代码来自https://my.oschina.net/sniperLi/blog/491854

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.1.xsd
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd">

    <!-- 激活组件扫描功能,在包com.hdj.learn.spring.aop及其子包下面自动扫描通过注解配置的组件 -->
    <context:component-scan base-package="com.hdj.learn.spring.aop"/>
    <!-- 激活自动代理功能 -->
    <aop:aspectj-autoproxy proxy-target-class="true"/>

    <!-- 用户服务对象 -->
    <bean id="userService" class="com.hdj.learn.spring.aop.service.UserService" />

</beans>
//声明这是一个组件
@Component
//声明这是一个切面Bean
@Aspect
@Slf4j
public class ServiceAspect {

    //配置切入点,该方法无方法体,主要为方便同类中其他方法使用此处配置的切入点
    @Pointcut("execution(* com.hdj.learn.spring.aop.service..*(..))")
    public void aspect() {
    }

    /*
     * 配置前置通知,使用在方法aspect()上注册的切入点
     * 同时接受JoinPoint切入点对象,可以没有该参数
     */
    @Before("aspect()")
    public void before(JoinPoint joinPoint) {
        log.info("before " + joinPoint);
    }

    //配置后置通知,使用在方法aspect()上注册的切入点
    @After("aspect()")
    public void after(JoinPoint joinPoint) {
        log.info("after " + joinPoint);
    }

    //配置环绕通知,使用在方法aspect()上注册的切入点
    @Around("aspect()")
    public void around(JoinPoint joinPoint) {
        long start = System.currentTimeMillis();
        try {
            ((ProceedingJoinPoint) joinPoint).proceed();
            long end = System.currentTimeMillis();
            log.info("around " + joinPoint + "\tUse time : " + (end - start) + " ms!");
        } catch (Throwable e) {
            long end = System.currentTimeMillis();
            log.info("around " + joinPoint + "\tUse time : " + (end - start) + " ms with exception : " + e.getMessage());
        }
    }

    //配置后置返回通知,使用在方法aspect()上注册的切入点
    @AfterReturning("aspect()")
    public void afterReturn(JoinPoint joinPoint) {
        log.info("afterReturn " + joinPoint);
    }

    //配置抛出异常后通知,使用在方法aspect()上注册的切入点
    @AfterThrowing(pointcut = "aspect()", throwing = "ex")
    public void afterThrow(JoinPoint joinPoint, Exception ex) {
        log.info("afterThrow " + joinPoint + "\t" + ex.getMessage());
    }

}

通过xml 及 注解的方式,我们就完成了最简单的spring aop的配置

客户端调用

//client 调用
public static void main(String[] args) {
    ApplicationContext context = new ClassPathXmlApplicationContext("classpath:aop/applicationContext.xml");
    UserService person = (UserService) context.getBean("userService");
    person.getUser();
}

结果输出

img_283ccf6a355bb93d6a97c420f5869f9a.png

看这里,我们在不改变userService的方法前提下,往person.getUser的方法之前,及方法之后,添加了代码。

于是,我们会从以下几个角度来分析源码:
1)最重要的,如何在不改变源码的前提下,往代码的前后增加代码
2)如何配置要在哪个方法前后增加代码

这篇博客先进行一个概述,细节会在后面几篇博客里分析。

aop:aspectj-autoproxy 开启自动代理

img_f06a13717681e587d3425445ba67c456.png

之前分析 spring注解配置依赖注入时,我们有分析过annotation-config,aop和此处的处理方式一致,都是配置一个专门的解析器,对其进行解析。

img_512aaefdc4160284123e01ba4ca78a98.png

追一下调用过程,大概流程如下:
1)注册 internalAutoProxyCreator 的 实现类 AnnotationAwareAspectJAutoProxyCreator
2)如果配置了proxy-target-class 为true,将proxyTargetClass这个属性记录下来
3)expose-proxy 同样,如果存在,记录exposeProxy为true
4)注册该BeanDefinition

代理对象的创建

分析了spring aop的创建,继续分析spring aop

spring初始化到创建aop代理对象流程图

img_0594ffd5776fbada213fd874fb7627b3.png

cglib实现

由于该代理对象没有实现接口,所以这里底层使用cglib对其进行增强


img_7d16ff8576f9804c710bc81591d648cd.png

总结

有点事,今天就写到这里了2333,很多细节没有说清楚,后续会完善的。
后续会完善以下几点:
1)切面,通知的配置
2)方法增强原理
todo o(╥﹏╥)o

目录
相关文章
|
29天前
|
存储 缓存 Java
Spring高手之路23——AOP触发机制与代理逻辑的执行
本篇文章深入解析了Spring AOP代理的触发机制和执行流程,从源码角度详细讲解了Bean如何被AOP代理,包括代理对象的创建、配置与执行逻辑,帮助读者全面掌握Spring AOP的核心技术。
36 3
Spring高手之路23——AOP触发机制与代理逻辑的执行
|
2月前
|
设计模式 Java 测试技术
spring复习04,静态代理动态代理,AOP
这篇文章讲解了Java代理模式的相关知识,包括静态代理和动态代理(JDK动态代理和CGLIB),以及AOP(面向切面编程)的概念和在Spring框架中的应用。文章还提供了详细的示例代码,演示了如何使用Spring AOP进行方法增强和代理对象的创建。
spring复习04,静态代理动态代理,AOP
|
5月前
|
缓存 监控 安全
AOP则关注如何将那些影响多个类的功能模块化
AOP则关注如何将那些影响多个类的功能模块化
|
3月前
|
缓存 Java 开发者
Spring高手之路22——AOP切面类的封装与解析
本篇文章深入解析了Spring AOP的工作机制,包括Advisor和TargetSource的构建与作用。通过详尽的源码分析和实际案例,帮助开发者全面理解AOP的核心技术,提升在实际项目中的应用能力。
45 0
Spring高手之路22——AOP切面类的封装与解析
|
3月前
|
缓存 安全 Java
Spring AOP 中两种代理类型的限制
【8月更文挑战第22天】
31 0
|
3月前
|
Java Spring
|
4月前
|
缓存 安全 Java
Spring高手之路21——深入剖析Spring AOP代理对象的创建
本文详细介绍了Spring AOP代理对象的创建过程,分为三个核心步骤:判断是否增强、匹配增强器和创建代理对象。通过源码分析和时序图展示,深入剖析了Spring AOP的工作原理,帮助读者全面理解Spring AOP代理对象的生成机制及其实现细节。
48 0
Spring高手之路21——深入剖析Spring AOP代理对象的创建
|
6月前
|
设计模式 安全 Java
深入理解Spring Boot AOP:CGLIB代理与JDK动态代理的完全指南
深入理解Spring Boot AOP:CGLIB代理与JDK动态代理的完全指南
1764 1
|
6月前
|
设计模式 Java Spring
Spring AOP基础之代理模式.静态代理和动态代理
Spring AOP基础之代理模式.静态代理和动态代理
67 0
Spring AOP基础之代理模式.静态代理和动态代理
|
6月前
|
缓存 Java uml
SpringBoot2 | Spring AOP 原理深度源码分析(八)
SpringBoot2 | Spring AOP 原理深度源码分析(八)
118 0