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

相关文章
|
1天前
|
NoSQL Java Redis
Spring Boot 自动配置机制:从原理到自定义
Spring Boot 的自动配置机制通过 `spring.factories` 文件和 `@EnableAutoConfiguration` 注解,根据类路径中的依赖和条件注解自动配置所需的 Bean,大大简化了开发过程。本文深入探讨了自动配置的原理、条件化配置、自定义自动配置以及实际应用案例,帮助开发者更好地理解和利用这一强大特性。
30 14
|
18天前
|
监控 安全 Java
什么是AOP?如何与Spring Boot一起使用?
什么是AOP?如何与Spring Boot一起使用?
44 5
|
23天前
|
Java 开发者 Spring
深入解析:Spring AOP的底层实现机制
在现代软件开发中,Spring框架的AOP(面向切面编程)功能因其能够有效分离横切关注点(如日志记录、事务管理等)而备受青睐。本文将深入探讨Spring AOP的底层原理,揭示其如何通过动态代理技术实现方法的增强。
50 8
|
23天前
|
Java 开发者 Spring
Spring AOP 底层原理技术分享
Spring AOP(面向切面编程)是Spring框架中一个强大的功能,它允许开发者在不修改业务逻辑代码的情况下,增加额外的功能,如日志记录、事务管理等。本文将深入探讨Spring AOP的底层原理,包括其核心概念、实现方式以及如何与Spring框架协同工作。
|
23天前
|
XML 监控 安全
深入调查研究Spring AOP
【11月更文挑战第15天】
36 5
|
23天前
|
Java 开发者 Spring
Spring AOP深度解析:探秘动态代理与增强逻辑
Spring框架中的AOP(Aspect-Oriented Programming,面向切面编程)功能为开发者提供了一种强大的工具,用以将横切关注点(如日志、事务管理等)与业务逻辑分离。本文将深入探讨Spring AOP的底层原理,包括动态代理机制和增强逻辑的实现。
34 4
|
28天前
|
安全 Java 开发者
AOP中的JDK动态代理与CGLIB动态代理:深度解析与实战模拟
【11月更文挑战第21天】面向切面编程(AOP,Aspect-Oriented Programming)是一种编程范式,它通过将横切关注点(cross-cutting concerns)与业务逻辑分离,以提高代码的可维护性和可重用性。在Java开发中,AOP的实现离不开动态代理技术,其中JDK动态代理和CGLIB动态代理是两种常用的方式。本文将从背景、历史、功能点、业务场景、底层逻辑等多个维度,深度解析这两种代理方式的区别,并通过Java示例进行模拟和比较。
44 4
|
7月前
|
安全 Java Spring
Spring之Aop的底层原理
Spring之Aop的底层原理
|
7月前
|
设计模式 Java uml
Spring AOP 原理
Spring AOP 原理
41 0
|
Java Spring 容器
【Spring AOP底层实现原理】
【Spring AOP底层实现原理】
165 0