浅谈基于动态代理的Spring AOP原理

简介: 浅谈基于动态代理的Spring AOP原理

浅谈基于动态代理的Spring AOP原理

1 什么时候会用到AOP?

  • Bean的创建和初始化过程中
  • 增加环绕、前置、后置方法
  • 事务、日志
  • 拦截器

这篇文章就着重讲解下Bean创建和初始化的过程中使用到的AOP原理

2 回顾Bean的创建过程

网络异常,图片无法展示
|


在Bean的创建过程中比较重要的一步就是对Bean进行AOP处理,但是不是必要的,接下来我们从头开始手写一个自己的Spring进行实现。

3 扫描Bean

这里的流程主要是new一个ApplicationContext,传入xml文件或配置类,以此来扫描固定包下的类。

网络异常,图片无法展示
|


/**
 * 构造方法
 *
 * @param configClass
 */
public MyApplicationContext(Class configClass) {
    this.configClass = configClass;
    //扫描Bean
    scanBean();
    //根据BeanDefinition创建Bean
    for (Map.Entry<String, MyBeanDefinition> beanDefinitionEntry : beanDefinitionMap.entrySet()) {
        String key = beanDefinitionEntry.getKey();
        MyBeanDefinition value = beanDefinitionEntry.getValue();
        if (ScopeUtils.SINGLETON.equals(value.getScope())) {
            Object bean = createBean(key,value);
            singletonPool.put(key, bean);
        }
    }
}
/**
 * 扫描Bean
 */
private void scanBean() {
    //拿到配置类之后对其进行解析 扫描
    MyComponentScan scan = (MyComponentScan) configClass.getDeclaredAnnotation(MyComponentScan.class);
    String path = scan.value();
    //根据路径进行扫描
    ClassLoader classLoader = MyComponentScan.class.getClassLoader();
    String newPath = path.replace('.', '/');
    //获取文件路径
    URL url = classLoader.getResource(newPath);
    File file = new File(url.getFile());
    if (file.isDirectory()) {
        File[] files = file.listFiles();
        //获取到该包下的每个文件(Class)
        for (File f : files) {
            String absolutePath = f.getAbsolutePath();
            //判断是不是.class文件
            if (absolutePath.endsWith(".class")) {
                String className = absolutePath.substring(absolutePath.indexOf("org"), absolutePath.indexOf(".class"));
                String name = className.replace("\\", ".");
                try {
                    Class<?> clazz = classLoader.loadClass(name);
                    //判断是否有MyComponent注解
                    if (clazz.isAnnotationPresent(MyComponent.class)) {
                        MyComponent component = clazz.getDeclaredAnnotation(MyComponent.class);
                        String beanName = component.value();
                        if (beanName.getBytes().length == 0) {
                            beanName = name.substring(name.lastIndexOf(".") + 1);
                            beanName = beanName.substring(0, 1).toLowerCase(Locale.ROOT) + beanName.substring(1);
                        }
                        MyBeanDefinition beanDefinition = new MyBeanDefinition();
                        beanDefinition.setClazz(clazz);
                        //判断bean的作用域
                        if (clazz.isAnnotationPresent(MyScope.class)) {
                            MyScope scope = clazz.getDeclaredAnnotation(MyScope.class);
                            beanDefinition.setScope(scope.value());
                        } else {
                            beanDefinition.setScope("singleton");
                        }
                        //判断是否实现了BeanPostProcessor
                        if (MyBeanPostProcessor.class.isAssignableFrom(clazz)) {
                            //这一步简化,非spring源码
                            MyBeanPostProcessor instance = (MyBeanPostProcessor) clazz.getDeclaredConstructor().newInstance();
                            beanPostProcessorList.add(instance);
                        }
                        //将该bean的Definition放入map中
                        beanDefinitionMap.put(beanName, beanDefinition);
                    }
                } catch (ClassNotFoundException e) {
                    e.printStackTrace();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
复制代码

4 创建Bean

网络异常,图片无法展示
|


/**
 * 创建bean
 *
 * @param beanDefinition
 * @return
 */
public Object createBean(String beanName,MyBeanDefinition beanDefinition) {
    Class clazz = beanDefinition.getClazz();
    Object newInstance = null;
    try {
        newInstance = clazz.getConstructor().newInstance();
        //处理依赖注入
        for (Field field : clazz.getDeclaredFields()) {
            if (field.isAnnotationPresent(MyAutowired.class)) {
                Object bean = getBean(field.getName());
                field.setAccessible(true);
                field.set(newInstance, bean);
            }
        }
        //BeanPostProcessor初始化前
        for (MyBeanPostProcessor processor : beanPostProcessorList) {
            //放入的和传出的不一定是同一个对象
            newInstance = processor.postProcessBeforeInitialization(newInstance, beanName);
        }
        //Bean的初始化方法
        if (newInstance instanceof MyInitializingBean) {
            ((MyInitializingBean) newInstance).afterPropertiesSet();
        }
        //BeanPostProcessor初始化后
        for (MyBeanPostProcessor processor : beanPostProcessorList) {
            //放入的和传出的不一定是同一个对象
            newInstance = processor.postProcessAfterInitialization(newInstance, beanName);
        }
    } catch (InstantiationException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    } catch (InvocationTargetException e) {
        e.printStackTrace();
    } catch (NoSuchMethodException e) {
        e.printStackTrace();
    }
    return newInstance;
}
复制代码

5 获取Bean

网络异常,图片无法展示
|


/**
 * getBean方法
 *
 * @param beanName
 * @return
 */
public Object getBean(String beanName) {
    if (beanDefinitionMap.containsKey(beanName)) {
        MyBeanDefinition beanDefinition = beanDefinitionMap.get(beanName);
        if (ScopeUtils.SINGLETON.equals(beanDefinition.getScope())) {
            return singletonPool.get(beanName);
        } else {
            return createBean(beanName,beanDefinition);
        }
    } else {
        throw new MyNoSuchBeanDefinitionException("没有这个Bean");
    }
}
复制代码

6 实现自己的BeanPostProcessor

Spring初始化Bean时对Bean的增强就是在BeanPostProcess方法中进行实现的,postProcessBeforeInitialization做前置处理,postProcessAfterInitialization做后置处理。可以看下Spring源码

网络异常,图片无法展示
|


/**
 * @desc: BeanPostProcessor抽象
 * @author: YanMingXin
 * @create: 2021/9/16-15:01
 **/
public interface MyBeanPostProcessor {
    Object postProcessBeforeInitialization(Object bean,String beanName);
    Object postProcessAfterInitialization(Object bean,String beanName);
}
复制代码

实现该方法,使用JDK动态代理将Bean包装

网络异常,图片无法展示
|


/**
 * @desc: MyBeanPostProcessorHandler
 * @author: YanMingXin
 * @create: 2021/9/17-10:25
 **/
@MyComponent
public class MyBeanPostProcessorHandler implements MyBeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) {
        if ("studentServiceImpl".equals(beanName)) {
            System.out.println("初始化前");
        }
        return bean;
    }
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) {
        if ("studentServiceImpl".equals(beanName)) {
            Object proxyInstance = Proxy.newProxyInstance(MyBeanPostProcessorHandler.class.getClassLoader(), bean.getClass().getInterfaces(), new InvocationHandler() {
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                    //代理逻辑
                    System.out.println("代理逻辑");
                    //对象的方法
                    return method.invoke(bean, args);
                }
            });
            System.out.println("初始化后");
            return proxyInstance;
        }
        return bean;
    }
}
复制代码

7 测试

网络异常,图片无法展示
|



相关文章
|
4月前
|
监控 安全 Java
Spring AOP实现原理
本内容主要介绍了Spring AOP的核心概念、实现机制及代理生成流程。涵盖切面(Aspect)、连接点(Join Point)、通知(Advice)、切点(Pointcut)等关键概念,解析了JDK动态代理与CGLIB代理的原理及对比,并深入探讨了通知执行链路和责任链模式的应用。同时,详细分析了AspectJ注解驱动的AOP解析过程,包括切面识别、切点表达式匹配及通知适配为Advice的机制,帮助理解Spring AOP的工作原理与实现细节。
|
22天前
|
Java 关系型数据库 数据库
深度剖析【Spring】事务:万字详解,彻底掌握传播机制与事务原理
在Java开发中,Spring框架通过事务管理机制,帮我们轻松实现了这种“承诺”。它不仅封装了底层复杂的事务控制逻辑(比如手动开启、提交、回滚事务),还提供了灵活的配置方式,让开发者能专注于业务逻辑,而不用纠结于事务细节。
|
26天前
|
人工智能 监控 安全
如何快速上手【Spring AOP】?核心应用实战(上篇)
哈喽大家好吖~欢迎来到Spring AOP系列教程的上篇 - 应用篇。在本篇,我们将专注于Spring AOP的实际应用,通过具体的代码示例和场景分析,帮助大家掌握AOP的使用方法和技巧。而在后续的下篇中,我们将深入探讨Spring AOP的实现原理和底层机制。 AOP(Aspect-Oriented Programming,面向切面编程)是Spring框架中的核心特性之一,它能够帮助我们解决横切关注点(如日志记录、性能统计、安全控制、事务管理等)的问题,提高代码的模块化程度和复用性。
|
25天前
|
设计模式 Java 开发者
如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。
|
2月前
|
缓存 安全 Java
Spring 框架核心原理与实践解析
本文详解 Spring 框架核心知识,包括 IOC(容器管理对象)与 DI(容器注入依赖),以及通过注解(如 @Service、@Autowired)声明 Bean 和注入依赖的方式。阐述了 Bean 的线程安全(默认单例可能有安全问题,需业务避免共享状态或设为 prototype)、作用域(@Scope 注解,常用 singleton、prototype 等)及完整生命周期(实例化、依赖注入、初始化、销毁等步骤)。 解析了循环依赖的解决机制(三级缓存)、AOP 的概念(公共逻辑抽为切面)、底层动态代理(JDK 与 Cglib 的区别)及项目应用(如日志记录)。介绍了事务的实现(基于 AOP
|
2月前
|
监控 架构师 NoSQL
spring 状态机 的使用 + 原理 + 源码学习 (图解+秒懂+史上最全)
spring 状态机 的使用 + 原理 + 源码学习 (图解+秒懂+史上最全)
|
4月前
|
前端开发 Java 数据库连接
Spring核心原理剖析与解说
每个部分都是将一种巨大并且复杂的技术理念传达为更易于使用的接口,而这就是Spring的价值所在,它能让你专注于开发你的应用,而不必从头开始设计每一部分。
164 32
|
4月前
|
Java 开发者 Spring
Spring框架 - 深度揭秘Spring框架的基础架构与工作原理
所以,当你进入这个Spring的世界,看似一片混乱,但细看之下,你会发现这里有个牢固的结构支撑,一切皆有可能。不论你要建设的是一座宏大的城堡,还是个小巧的花园,只要你的工具箱里有Spring,你就能轻松搞定。
184 9
|
Java Spring
Spring5系列(十) | 动态代理底层实现
Spring5系列(十) | 动态代理底层实现
226 0
Spring5系列(十) | 动态代理底层实现
|
2月前
|
Java Spring 容器
SpringBoot自动配置的原理是什么?
Spring Boot自动配置核心在于@EnableAutoConfiguration注解,它通过@Import导入配置选择器,加载META-INF/spring.factories中定义的自动配置类。这些类根据@Conditional系列注解判断是否生效。但Spring Boot 3.0后已弃用spring.factories,改用新格式的.imports文件进行配置。
716 0