浅谈基于动态代理的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 测试

相关文章
|
20天前
|
XML Java 开发者
Spring Boot开箱即用可插拔实现过程演练与原理剖析
【11月更文挑战第20天】Spring Boot是一个基于Spring框架的项目,其设计目的是简化Spring应用的初始搭建以及开发过程。Spring Boot通过提供约定优于配置的理念,减少了大量的XML配置和手动设置,使得开发者能够更专注于业务逻辑的实现。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,为开发者提供一个全面的理解。
26 0
|
25天前
|
XML Java 数据安全/隐私保护
Spring Aop该如何使用
本文介绍了AOP(面向切面编程)的基本概念和术语,并通过具体业务场景演示了如何在Spring框架中使用Spring AOP。文章详细解释了切面、连接点、通知、切点等关键术语,并提供了完整的示例代码,帮助读者轻松理解和应用Spring AOP。
Spring Aop该如何使用
|
2月前
|
存储 缓存 Java
Spring高手之路23——AOP触发机制与代理逻辑的执行
本篇文章深入解析了Spring AOP代理的触发机制和执行流程,从源码角度详细讲解了Bean如何被AOP代理,包括代理对象的创建、配置与执行逻辑,帮助读者全面掌握Spring AOP的核心技术。
43 3
Spring高手之路23——AOP触发机制与代理逻辑的执行
|
1月前
|
Java Spring
[Spring]aop的配置与使用
本文介绍了AOP(面向切面编程)的基本概念和核心思想。AOP是Spring框架的核心功能之一,通过动态代理在不修改原代码的情况下注入新功能。文章详细解释了连接点、切入点、通知、切面等关键概念,并列举了前置通知、后置通知、最终通知、异常通知和环绕通知五种通知类型。
30 1
|
2月前
|
Java 数据安全/隐私保护 Spring
Spring进阶:初识动态代理
本文介绍了Spring框架中AOP切面编程的基础——动态代理。通过定义Vehicle接口及其实现类Car和Ship,展示了如何使用动态代理在不修改原代码的基础上增强功能。文章详细解释了动态代理的工作原理,包括通过`Proxy.newProxyInstance()`方法创建代理对象,以及`InvocationHandler`接口中的`invoke()`方法如何处理代理对象的方法调用。最后,通过一个测试类`TestVehicle`演示了动态代理的具体应用。
|
26天前
|
安全 Java 测试技术
Java开发必读,谈谈对Spring IOC与AOP的理解
Spring的IOC和AOP机制通过依赖注入和横切关注点的分离,大大提高了代码的模块化和可维护性。IOC使得对象的创建和管理变得灵活可控,降低了对象之间的耦合度;AOP则通过动态代理机制实现了横切关注点的集中管理,减少了重复代码。理解和掌握这两个核心概念,是高效使用Spring框架的关键。希望本文对你深入理解Spring的IOC和AOP有所帮助。
33 0
|
2月前
|
Java Spring 容器
Spring底层原理大致脉络
Spring底层原理大致脉络
|
2月前
|
Java 编译器 Spring
Spring AOP 和 AspectJ 的区别
Spring AOP和AspectJ AOP都是面向切面编程(AOP)的实现,但它们在实现方式、灵活性、依赖性、性能和使用场景等方面存在显著区别。‌
89 2
|
2月前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
【10月更文挑战第1天】Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
137 9
|
2月前
|
XML Java 数据格式
Spring的IOC和AOP
Spring的IOC和AOP
50 0