Spring扫描逻辑原码解析(带图好理解)

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: Spring扫描逻辑原码解析(带图好理解)

先上流程图

再上类图

再上代码

    public static void main(String[] args) {
        XsmApplicationContext applicationContext=new XsmApplicationContext(AppConfig.class);
        System.out.println(applicationContext.getBean("userService"));
        System.out.println(applicationContext.getBean("userService"));
        System.out.println(applicationContext.getBean("userService"));
    }
@ComponentScan("com.xiashiman.service")
public class AppConfig {}
@Component("userService")
//@Scope("prototype")
public class UserService {}
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface ComponentScan {
    String value();
}
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Component {
    String value();
}
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Scope {
    String value();
}
public class BeanDefinition {
    private Class clazz;
    private String scope;
    public BeanDefinition(Class clazz, String scope) {
        this.clazz = clazz;
        this.scope = scope;
    }
    public BeanDefinition() {
    }
    public Class getClazz() {
        return clazz;
    }
    public void setClazz(Class clazz) {
        this.clazz = clazz;
    }
    public String getScope() {
        return scope;
    }
    public void setScope(String scope) {
        this.scope = scope;
    }
}
public class XsmApplicationContext {
    private  Class configClass;
    private ConcurrentHashMap<String,Object> singletonObjects=new ConcurrentHashMap<>();//单例池
    private ConcurrentHashMap<String,BeanDefinition> beanDefinitionMap=new ConcurrentHashMap<>();
    public XsmApplicationContext(Class configClass) {
        this.configClass = configClass;
        //ComponentScan注解--->扫描路径---》扫描---》Beandefinition--->BeanDefinitionMap
        scan(configClass);
        //在启动时把所有单例bean创建好
        for(Map.Entry<String,BeanDefinition> entry:beanDefinitionMap.entrySet()){
            String beanName=entry.getKey();
            BeanDefinition beanDefinition=entry.getValue();
            if(beanDefinition.getScope().equals("singleton")){
                Object bean= createBean(beanDefinition); //单例bean
                singletonObjects.put(beanName,bean);
            }
        }
    }
    //根据bean的生命周期创建对象
    public Object createBean(BeanDefinition beanDefinition){
        Class clazz = beanDefinition.getClazz();
        try {
            Object instance = clazz.getDeclaredConstructor().newInstance();
            return instance;//返回实例对象
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        }
        return null;
    }
    private void scan(Class configClass) {
        if(configClass.isAnnotationPresent(ComponentScan.class)){
            //解析配置类
            //ComponentScan注解--->扫描路径---》扫描
            ComponentScan declaredAnnotation = (ComponentScan) configClass.getDeclaredAnnotation(ComponentScan.class);
            String path = declaredAnnotation.value(); //扫描路径 com.xiashiman.service
            path=path.replace(".","/"); //把·替换成/---》 com/xiashiman/service
            //扫描--->找出包下面的所有类
            //类加载器
            //Bootstrap----jre/lib
            //Ext----->jre/ext/lib
            //App----->classpath-----:E:\project\common\shousiCode\target\classes
            ClassLoader classLoader=XsmApplicationContext.class.getClassLoader();//app类加载器
            URL resource = classLoader.getResource(path);//通过路径获取资源
            File file=new File(resource.getFile());
            //判断是否是目录
            if(file.isDirectory()){
                File[] files = file.listFiles();//拿到目录下的所以文件
                for (File f:files){
                    String fileName = f.getAbsolutePath(); //E:\project\common\shousiCode\target\classes\com\xiashiman\service\UserService.class
                    //判断是否是class文件
                    if(fileName.endsWith(".class")){
                        //截取com到.class前的部分
                        String className= fileName.substring(fileName.indexOf("com\\"), fileName.indexOf(".class"));
                        className=className.replace("\\",".");  //替换成:com.xiashiman.service.UserService
                        //准备通过反射实例化对象
                        Class clazz=null;
                        try{
                            clazz=classLoader.loadClass(className); //反射获取类模板对象
                            if(clazz.isAnnotationPresent(Component.class)){
                                //表示当前类是一个bean
                                //解析类:判断是单例 还是原型
                                // BeanDefinition
                                Component componentAnntotaion= (Component) clazz.getDeclaredAnnotation(Component.class);
                                String beanName = componentAnntotaion.value();
                                BeanDefinition beanDefinition=new BeanDefinition();
                                beanDefinition.setClazz(clazz);
                                if(clazz.isAnnotationPresent(Scope.class)){
                                    Scope scopeAnntation= (Scope) clazz.getDeclaredAnnotation(Scope.class);
                                    beanDefinition.setScope(scopeAnntation.value());
                                }else{
                                    beanDefinition.setScope("singleton");
                                }
                                beanDefinitionMap.put(beanName,beanDefinition);
                            }
                        }catch (ClassNotFoundException e){
                            e.printStackTrace();
                        }
                    }
                }
            }
        }
    }
    public Object getBean(String beanName){
        if(beanDefinitionMap.containsKey(beanName)){
            BeanDefinition beanDefinition=beanDefinitionMap.get(beanName);
            if(beanDefinition.getScope().equals("singleton")){
                Object o=singletonObjects.get(beanName);
                return o;
            }else{
                //创建bean对象  怎么创建呢
                Object bean=createBean(beanDefinition);
                return bean;
            }
        }else{
            throw new NullPointerException("不存在对应的bean");
        }
    }
}
相关文章
|
8天前
|
Java Spring
【Spring】方法注解@Bean,配置类扫描路径
@Bean方法注解,如何在同一个类下面定义多个Bean对象,配置扫描路径
132 73
|
9天前
|
设计模式 XML Java
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
本文详细介绍了Spring框架的核心功能,并通过手写自定义Spring框架的方式,深入理解了Spring的IOC(控制反转)和DI(依赖注入)功能,并且学会实际运用设计模式到真实开发中。
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
|
2月前
|
前端开发 Java 开发者
Spring MVC中的请求映射:@RequestMapping注解深度解析
在Spring MVC框架中,`@RequestMapping`注解是实现请求映射的关键,它将HTTP请求映射到相应的处理器方法上。本文将深入探讨`@RequestMapping`注解的工作原理、使用方法以及最佳实践,为开发者提供一份详尽的技术干货。
135 2
|
2月前
|
前端开发 Java Spring
探索Spring MVC:@Controller注解的全面解析
在Spring MVC框架中,`@Controller`注解是构建Web应用程序的基石之一。它不仅简化了控制器的定义,还提供了一种优雅的方式来处理HTTP请求。本文将全面解析`@Controller`注解,包括其定义、用法、以及在Spring MVC中的作用。
57 2
|
2月前
|
前端开发 Java Maven
深入解析:如何用 Spring Boot 实现分页和排序
深入解析:如何用 Spring Boot 实现分页和排序
59 2
|
2月前
|
Java 开发者 Spring
Spring AOP深度解析:探秘动态代理与增强逻辑
Spring框架中的AOP(Aspect-Oriented Programming,面向切面编程)功能为开发者提供了一种强大的工具,用以将横切关注点(如日志、事务管理等)与业务逻辑分离。本文将深入探讨Spring AOP的底层原理,包括动态代理机制和增强逻辑的实现。
48 4
|
2月前
|
前端开发 Java 开发者
Spring MVC中的控制器:@Controller注解全解析
在Spring MVC框架中,`@Controller`注解是构建Web应用程序控制层的核心。它不仅简化了控制器的定义,还提供了灵活的请求映射和处理机制。本文将深入探讨`@Controller`注解的用法、特点以及在实际开发中的应用。
96 0
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
86 2
|
9天前
|
存储 设计模式 算法
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务,它涉及算法与对象间职责的分配。行为型模式分为类行为模式和对象行为模式,前者采用继承机制来在类间分派行为,后者采用组合或聚合在对象间分配行为。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象行为模式比类行为模式具有更大的灵活性。 行为型模式分为: • 模板方法模式 • 策略模式 • 命令模式 • 职责链模式 • 状态模式 • 观察者模式 • 中介者模式 • 迭代器模式 • 访问者模式 • 备忘录模式 • 解释器模式
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
|
9天前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
结构型模式描述如何将类或对象按某种布局组成更大的结构。它分为类结构型模式和对象结构型模式,前者采用继承机制来组织接口和类,后者釆用组合或聚合来组合对象。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象结构型模式比类结构型模式具有更大的灵活性。 结构型模式分为以下 7 种: • 代理模式 • 适配器模式 • 装饰者模式 • 桥接模式 • 外观模式 • 组合模式 • 享元模式
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析

推荐镜像

更多