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

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 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");
        }
    }
}
相关文章
|
3天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
14 2
|
21天前
|
搜索推荐 Java Spring
Spring Filter深度解析
【10月更文挑战第21天】Spring Filter 是 Spring 框架中非常重要的一部分,它为请求处理提供了灵活的控制和扩展机制。通过合理配置和使用 Filter,可以实现各种个性化的功能,提升应用的安全性、可靠性和性能。还可以结合具体的代码示例和实际应用案例,进一步深入探讨 Spring Filter 的具体应用和优化技巧,使对它的理解更加全面和深入。
|
28天前
|
存储 缓存 Java
Spring高手之路23——AOP触发机制与代理逻辑的执行
本篇文章深入解析了Spring AOP代理的触发机制和执行流程,从源码角度详细讲解了Bean如何被AOP代理,包括代理对象的创建、配置与执行逻辑,帮助读者全面掌握Spring AOP的核心技术。
35 3
Spring高手之路23——AOP触发机制与代理逻辑的执行
|
1月前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
108 5
|
1月前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
30天前
|
XML Java 数据格式
Spring IOC容器的深度解析及实战应用
【10月更文挑战第14天】在软件工程中,随着系统规模的扩大,对象间的依赖关系变得越来越复杂,这导致了系统的高耦合度,增加了开发和维护的难度。为解决这一问题,Michael Mattson在1996年提出了IOC(Inversion of Control,控制反转)理论,旨在降低对象间的耦合度,提高系统的灵活性和可维护性。Spring框架正是基于这一理论,通过IOC容器实现了对象间的依赖注入和生命周期管理。
65 0
|
1月前
|
XML Java 数据格式
手动开发-简单的Spring基于注解配置的程序--源码解析
手动开发-简单的Spring基于注解配置的程序--源码解析
46 0
|
1月前
|
XML Java 数据格式
手动开发-简单的Spring基于XML配置的程序--源码解析
手动开发-简单的Spring基于XML配置的程序--源码解析
79 0
|
1月前
|
缓存 Java 程序员
Map - LinkedHashSet&Map源码解析
Map - LinkedHashSet&Map源码解析
66 0
|
1月前
|
算法 Java 容器
Map - HashSet & HashMap 源码解析
Map - HashSet & HashMap 源码解析
52 0

推荐镜像

更多