SpringBoot之Class资源加载顺序

问题描述与猜想

在工作中遇到一个问题,代码和结构大致如下

如图所示,大致是一个目录下有A和B两个类,其中A类用@Bean注解实例化一个treeA的对象,B类使用@ConditionalOnBean注解判断spring是否有treeA对象,有的话才实例化,没有则跳过

在本地环境一直是没有问题的,但是当打成jar包放到测试环境中时,发现B类没有实例化,对此引发了我的探究。

我们知道,SpringBoot会先将带有@Compoment的类扫描出来,然后按顺序遍历,如果有条件注解则判断其是否符合条件,不符合直接跳过;符合则会将其类下使用@Import、@Bean等注解引入的Bean注册到bdm中。

上面两个类都用了@Configuration注解,肯定被扫描到了,所以可以肯定问题的根源是先加载了B类,发现没有treeA对象,所以跳过,然后再加载了A类。

在源代码中一番debug后,发现SpringBoot对于不同资源有不同的扫描实现,开发环境一般是target,是一个文件包,测试环境则是jar包。

scss 复制代码
PathMatchingResourcePatternResolver->findPathMatchingResources():
    if (rootDirUrl.getProtocol().startsWith(ResourceUtils.URL_PROTOCOL_VFS)) {
        result.addAll(VfsResourceMatchingDelegate.findMatchingResources(rootDirUrl, subPattern, getPathMatcher()));
    }
    else if (ResourceUtils.isJarURL(rootDirUrl) || isJarResource(rootDirResource)) {
        // jar实现
        result.addAll(doFindPathMatchingJarResources(rootDirResource, rootDirUrl, subPattern));
    }
    else {
        // target实现
        result.addAll(doFindPathMatchingFileResources(rootDirResource, subPattern));
    }

target实现

scss 复制代码
PathMatchingResourcePatternResolver->doFindPathMatchingFileResources():
    ......
    return doFindMatchingFileSystemResources(rootDir, subPattern);
PathMatchingResourcePatternResolver->doFindMatchingFileSystemResources():
    ......
    Set<File> matchingFiles = retrieveMatchingFiles(rootDir, subPattern);
PathMatchingResourcePatternResolver->retrieveMatchingFiles():
    ......
    doRetrieveMatchingFiles(fullPattern, rootDir, result);
PathMatchingResourcePatternResolver->doRetrieveMatchingFiles():
    for (File content : listDirectory(dir)) {
        ......
    }

重点在listDirectory(dir)这个方法中

ini 复制代码
PathMatchingResourcePatternResolver->listDirectory(File dir):
    File[] files = dir.listFiles();
    // 对于files是空的处理
    if (files == null) {
        ......
    }
    Arrays.sort(files, Comparator.comparing(File::getName));
    return files;

dir.listFiles(),列举目录中的所有文件,但不包含空目录。也就是说,如果有目录aa中有文件b.txt,则会把这个b.txt列举出来(无论有多少层级都会列出)。

Arrays.sort(files, Comparator.comparing(File::getName)),对文件名进行字典顺序排序,先数字,再大写字母,最后小写字母。

在开发环境中,A类总是排在B类的前面,所以不会出现B加载不到的问题。

jar实现

scss 复制代码
PathMatchingResourcePatternResolver->doFindPathMatchingJarResources():
    ......
    // 遍历文件
    for (Enumeration<JarEntry> entries = jarFile.entries(); entries.hasMoreElements();) {
        JarEntry entry = entries.nextElement();
        ......
    }
JarFile->JarEntryIterator->nextElement():
    return next();
JarFile->JarEntryIterator->next():
    ZipEntry ze = e.nextElement();
    return new JarFileEntry(ze);

注意到这里的e是ZipEntry的子类,于是找ZipEntry的nextElement方法

scss 复制代码
ZipFile->ZipEntryIterator->nextElement():
    return next();
ZipFile->ZipEntryIterator->next():
    ......
    long jzentry = getNextEntry(jzfile, i++);
    ......

ZipEntry的底层实现都是native方法,debug到这里戛然而止了,但是得出了结论,jar遍历资源文件底层走的是ZipEntry的遍历方法。

而我们知道,ZipEntry遍历顺序是根据假如中央字典的顺序来的,与文件大小、修改时间、文件字典名都没有关系

在linux中,使用zip -l 目录名的命令可以将zip包的中央字典顺序排列出来,执行上述命令后,发现果然B.class排在了A.class的前面,这也就解释通了。

解决方法

在介绍OnBeanCondition的时候总结过,建议条件注解只判断@Component层面是否有相应的bean,在项目中发现A类和B类都是必须的,没有必须先有A类才能有B类的前提,因此将B类的条件注解去掉,即可解决问题

相关推荐
JZC_xiaozhong8 分钟前
多系统权限标准不统一?企业如何实现跨平台统一权限管控
java·大数据·微服务·数据集成与应用集成·iam系统·权限治理·统一权限管理
爬山算法1 小时前
Hibernate(85)如何在持续集成/持续部署(CI/CD)中使用Hibernate?
java·ci/cd·hibernate
菜鸟233号2 小时前
力扣647 回文子串 java实现
java·数据结构·leetcode·动态规划
Charlie_lll2 小时前
力扣解题-[3379]转换数组
数据结构·后端·算法·leetcode
qq_12498707532 小时前
基于Java Web的城市花园小区维修管理系统的设计与实现(源码+论文+部署+安装)
java·开发语言·前端·spring boot·spring·毕业设计·计算机毕业设计
h7ml2 小时前
查券返利机器人的OCR识别集成:Java Tesseract+OpenCV优化图片验证码的自动解析方案
java·机器人·ocr
野犬寒鸦2 小时前
从零起步学习并发编程 || 第五章:悲观锁与乐观锁的思想与实现及实战应用与问题
java·服务器·数据库·学习·语言模型
Volunteer Technology2 小时前
Sentinel的限流算法
java·python·算法
VX:Fegn08952 小时前
计算机毕业设计|基于springboot + vue云租车平台系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
岁岁种桃花儿2 小时前
SpringCloud从入门到上天:Nacos做微服务注册中心
java·spring cloud·微服务