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

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



相关文章
|
6天前
|
运维 Java 程序员
Spring5深入浅出篇:基于注解实现的AOP
# Spring5 AOP 深入理解:注解实现 本文介绍了基于注解的AOP编程步骤,包括原始对象、额外功能、切点和组装切面。步骤1-3旨在构建切面,与传统AOP相似。示例代码展示了如何使用`@Around`定义切面和执行逻辑。配置中,通过`@Aspect`和`@Around`注解定义切点,并在Spring配置中启用AOP自动代理。 进一步讨论了切点复用,避免重复代码以提高代码维护性。通过`@Pointcut`定义通用切点表达式,然后在多个通知中引用。此外,解释了AOP底层实现的两种动态代理方式:JDK动态代理和Cglib字节码增强,默认使用JDK,可通过配置切换到Cglib
|
1天前
|
监控 安全 Java
Spring cloud原理详解
Spring cloud原理详解
11 0
|
6天前
|
XML Java 数据格式
Spring使用AOP 的其他方式
Spring使用AOP 的其他方式
15 2
|
6天前
|
XML Java 数据格式
Spring 项目如何使用AOP
Spring 项目如何使用AOP
19 2
|
7天前
|
Java 开发者 微服务
Spring Cloud原理详解
【5月更文挑战第4天】Spring Cloud是Spring生态系统中的微服务框架,包含配置管理、服务发现、断路器、API网关等工具,简化分布式系统开发。核心组件如Eureka(服务发现)、Config Server(配置中心)、Ribbon(负载均衡)、Hystrix(断路器)、Zuul(API网关)等。本文讨论了Spring Cloud的基本概念、核心组件、常见问题及解决策略,并提供代码示例,帮助开发者更好地理解和实践微服务架构。此外,还涵盖了服务通信方式、安全性、性能优化、自动化部署、服务网格和无服务器架构的融合等话题,揭示了微服务架构的未来趋势。
32 6
|
11天前
|
Java 开发者 Spring
Spring AOP的切点是通过使用AspectJ的切点表达式语言来定义的。
【5月更文挑战第1天】Spring AOP的切点是通过使用AspectJ的切点表达式语言来定义的。
23 5
|
11天前
|
XML Java 数据格式
Spring AOP
【5月更文挑战第1天】Spring AOP
27 5
|
12天前
|
Java 编译器 开发者
Spring的AOP理解
Spring的AOP理解
|
12天前
|
XML Java 数据格式
如何在Spring AOP中定义和应用通知?
【4月更文挑战第30天】如何在Spring AOP中定义和应用通知?
17 0
|
12天前
|
安全 Java 开发者
在Spring框架中,IoC和AOP是如何实现的?
【4月更文挑战第30天】在Spring框架中,IoC和AOP是如何实现的?
22 0