解读SpringBoot为什么要用DeferredImportSelector

 更新时间:2025年03月12日 08:39:16   作者:冰糖心书房  
这篇文章主要介绍了SpringBoot为什么要用DeferredImportSelector的情况,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

DeferredImportSelector 是 Spring Framework 3.1 引入的 ImportSelector 接口的增强版本,主要用于延迟加载配置类,提升 Spring Boot 应用的启动速度。

它允许在所有配置类处理完毕后,再根据其他配置类的信息选择性地导入一些配置类。

为什么使用 DeferredImportSelector?

传统的 ImportSelector 会在配置类解析的早期阶段执行,这意味着在选择要导入的配置类时,它可能无法访问完整的配置信息,导致无法做出最佳选择。

DeferredImportSelector 解决了这个问题,它的执行会被延迟到所有常规配置类都被处理之后,此时它可以访问到所有配置类的信息,从而做出更明智的导入决策。

DeferredImportSelector 的工作原理

注册: 当 Spring 容器解析配置类时,如果发现一个类实现了 DeferredImportSelector 接口,它不会立即执行 selectImports() 方法,而是将该 DeferredImportSelector 的实例注册到一个 DeferredImportSelectorHandler 中。

延迟执行: 在所有配置类都处理完毕后,DeferredImportSelectorHandler 会负责执行所有注册的 DeferredImportSelector

分组和排序 (可选): DeferredImportSelector 可以实现 DeferredImportSelector.Group 接口,将多个 DeferredImportSelector 分组,并按组进行排序。 Spring Boot 允许对 DeferredImportSelector.Group 进行排序以控制执行顺序。

selectImports() 方法:selectImports() 方法中,DeferredImportSelector 可以访问到整个 Spring 容器的配置信息,例如:

  • 已经注册的 Bean 定义。
  • 已经加载的配置类。
  • 环境信息 (Environment)。

返回要导入的配置类: selectImports() 方法返回一个 String 数组,包含了要导入的配置类的全限定名。 Spring 容器会根据这些全限定名来导入相应的配置类,并注册 Bean。

DeferredImportSelector 的优势

  • 延迟加载: 避免过早加载不必要的配置类,提升启动速度。
  • 更智能的配置选择: 基于已加载的配置信息,做出更精确的配置选择。
  • 条件化配置: 实现更复杂的条件化配置逻辑。
  • 排序和分组: 控制 DeferredImportSelector 的执行顺序。

DeferredImportSelector 的劣势

  • 更复杂:ImportSelector 更复杂,需要理解延迟加载和分组排序的机制。
  • 调试难度增加: 由于是延迟执行,调试时可能需要更多步骤。

DeferredImportSelector 的使用场景

  • 自动配置: Spring Boot 的自动配置机制大量使用了 DeferredImportSelector。 例如, EnableAutoConfiguration 注解使用 AutoConfigurationImportSelector 来选择需要导入的自动配置类。
  • 条件化配置: 根据某个 Bean 是否存在,或者某个配置属性的值来选择是否导入某个配置类。
  • 模块化: 将应用分解为多个模块,根据已加载的模块信息来选择导入其他模块的配置。

DeferredImportSelector 的示例

import org.springframework.context.annotation.DeferredImportSelector;
import org.springframework.core.type.AnnotationMetadata;

public class MyDeferredImportSelector implements DeferredImportSelector {

    @Override
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        // 可以获取注解信息 importingClassMetadata
        // 可以访问 Spring 容器中的 Bean 定义

        // 根据条件选择导入不同的配置类
        if (conditionA()) {
            return new String[] { "com.example.config.ConfigA" };
        } else {
            return new String[] { "com.example.config.ConfigB" };
        }
    }

    private boolean conditionA() {
        // 实现条件逻辑
        return true; // 示例:始终返回 true
    }
}

示例:分组和排序

import org.springframework.context.annotation.DeferredImportSelector;
import org.springframework.core.type.AnnotationMetadata;

import java.util.ArrayList;
import java.util.List;
import java.util.Objects;

public class MyDeferredImportSelector implements DeferredImportSelector {

    @Override
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        return new String[]{"com.example.MyConfig"};
    }

    @Override
    public Class<? extends Group> getImportGroup() {
        return MyGroup.class;
    }

    public static class MyGroup implements Group {

        private final List<Entry> entries = new ArrayList<>();

        @Override
        public void process(AnnotationMetadata metadata, DeferredImportSelector importSelector) {
            // 可以根据metadata和importSelector的信息来决定是否添加entry
            entries.add(new Entry(metadata, importSelector.selectImports(metadata)));
        }

        @Override
        public Iterable<Entry> selectImports() {
            // 返回需要导入的类的列表,这里可以对entries进行排序,控制导入的顺序
            return entries;
        }
    }
}

如何使用 DeferredImportSelector

  • 实现 DeferredImportSelector 接口: 创建一个类并实现 DeferredImportSelector 接口,并重写 selectImports() 方法。
  • 实现 getImportGroup() 方法 (可选): 如果需要分组和排序,则实现 getImportGroup() 并返回一个实现了 Group 接口的类。
  • 注册 DeferredImportSelector: 可以通过 @Import 注解,或者在 META-INF/spring.factories 文件中注册 DeferredImportSelector

Spring Boot 自动配置中的应用

Spring Boot 的自动配置机制利用 DeferredImportSelector 实现了延迟加载和条件化配置。

  • @EnableAutoConfiguration 注解: 这个注解触发了自动配置机制。
  • AutoConfigurationImportSelector: @EnableAutoConfiguration 最终会使用 AutoConfigurationImportSelector 这个 DeferredImportSelector
  • spring.factories 文件: AutoConfigurationImportSelectorMETA-INF/spring.factories 文件中读取 org.springframework.boot.autoconfigure.EnableAutoConfiguration 键对应的自动配置类列表。
  • 条件化过滤: AutoConfigurationImportSelector 会根据 Spring 容器中的条件 (例如,是否存在某个 Bean,或者某个配置属性的值) 来过滤这些自动配置类,只选择符合条件的自动配置类进行导入。

总结

DeferredImportSelector 是 Spring Boot 中一个重要的特性,它允许延迟加载和条件化配置,从而提升应用的启动速度和灵活性。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • Java 字节数组类型(byte[])与int类型互转方法

    Java 字节数组类型(byte[])与int类型互转方法

    下面小编就为大家带来一篇Java 字节数组类型(byte[])与int类型互转方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-02-02
  • mybatis update set 多个字段实例

    mybatis update set 多个字段实例

    这篇文章主要介绍了mybatis update set 多个字段实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-01-01
  • Java设置PDF有序和无序列表的知识点总结

    Java设置PDF有序和无序列表的知识点总结

    在本篇文章中小编给大家整理了关于Java设置PDF有序和无序列表的知识点,需要的朋友们参考下。
    2019-03-03
  • json-lib将json格式的字符串,转化为java对象的实例

    json-lib将json格式的字符串,转化为java对象的实例

    下面小编就为大家带来一篇json-lib将json格式的字符串,转化为java对象的实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-03-03
  • Java使用Tessdata做OCR图片文字识别的详细思路

    Java使用Tessdata做OCR图片文字识别的详细思路

    这篇文章主要介绍了Java使用Tessdata做OCR图片文字识别的详细思路,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-07-07
  • 使用springboot单元测试对weblistener的加载测试

    使用springboot单元测试对weblistener的加载测试

    这篇文章主要介绍了使用springboot单元测试对weblistener的加载测试,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • 详解Java多线程与并发

    详解Java多线程与并发

    多线程是一个进程在执行过程中产生多个更小的程序单元,这些更小的单元称为线程,这些线程可以同时存在,同时运行,一个进程可能包含多个同时执行的线程。多线程是实现并发机制的一种有效手段。进程和线程一样,都是实现并发的一个基本单位。
    2021-06-06
  • Java实现医院管理系统

    Java实现医院管理系统

    这篇文章主要介为大家详细绍了Java实现医院管理系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12
  • java调用ffmpeg实现转换视频

    java调用ffmpeg实现转换视频

    这篇文章主要为大家详细介绍了java调用ffmpeg实现转换视频功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12
  • java中hasNextInt判断后无限循环输出else项的解决方法

    java中hasNextInt判断后无限循环输出else项的解决方法

    这篇文章主要介绍了java中hasNextInt判断后无限循环输出else项的解决方法的相关资料,需要的朋友可以参考下
    2016-10-10

最新评论