Spring中@import注解终极揭秘!

Spring中@import注解终极揭秘 - 程序员古德

技术概念

它能干啥

@Import注解在Spring框架中主要用于解决模块化和配置管理方面的技术问题,它可以帮助开发者实现以下几个目标:

  1. 模块化配置:在大型项目中,通常需要将配置信息分散到多个配置类中,以便更好地组织和管理,@Import注解允许开发者在一个主配置类中导入其他配置类,从而实现配置的模块化。
  2. 第三方库或组件的集成:当项目中需要集成第三方库或组件时,这些库或组件可能会提供自己的配置类,通过@Import注解,开发者可以轻松地将这些第三方配置类集成到项目的总体配置中。
  3. 条件化配置@Import还可以与条件注解(如@Conditional)结合使用,以实现基于特定条件的配置加载,因此在不同的环境或情境下,可以加载不同的配置类,从而实现更加灵活和动态的配置管理。
  4. 扩展Spring功能:通过导入实现了特定接口的类,开发者可以扩展Spring框架的功能,比如,可以导入自定义的BeanFactoryPostProcessorBeanDefinitionRegistrar来修改或添加bean定义。
  5. 解决循环依赖问题:在某些情况下,使用@Import注解可以解决因循环依赖而导致的配置问题,通过将相互依赖的配置类分解并使用@Import进行导入,可以打破循环依赖的链条。

它有哪些特性

在Spring框架中,@Import注解可以用来引入一个或多个组件,这些组件通常是通过@Bean注解定义的,当使用@Import注解时,实际上是在告诉Spring:“除了当前配置类中的bean定义外,还想包含另一个配置类(或多个配置类)中定义的bean。”

@Import注解可以用来引入:

  1. 带有@Bean方法的配置类:这是最常见的情况,可以在一个配置类中定义bean,并使用@Import将其引入到其他配置类中。
  2. ImportSelector实现:这是一个更高级的特性,允许根据某些条件或运行时环境动态地选择要导入的配置类。
  3. ImportBeanDefinitionRegistrar实现:这是一个更底层的机制,允许在运行时手动注册bean定义。
  4. 使用@Import来组合多个配置类,从而构建复杂的配置层次结构。

使用@Import注解引入一个类

下面是一个简单的Java代码示例,演示了如何使用@Import注解来导入一个配置类。

首先,定义一个简单的服务接口GreetingService和其实现类GreetingServiceImpl

// GreetingService.java  
public interface GreetingService {  
    String sayGreeting();  
}  
  
// GreetingServiceImpl.java  
public class GreetingServiceImpl implements GreetingService {  
    @Override  
    public String sayGreeting() {  
        return "Hello, World!";  
    }  
}

接着,创建一个配置类GreetingConfig,该类使用@Bean注解来定义GreetingService的bean:

// GreetingConfig.java  
import org.springframework.context.annotation.Bean;  
import org.springframework.context.annotation.Configuration;  
  
@Configuration  
public class GreetingConfig {  
  
    @Bean  
    public GreetingService greetingService() {  
        return new GreetingServiceImpl();  
    }  
}

接着,创建一个主配置类AppConfig,并使用@Import注解来导入GreetingConfig类:

// AppConfig.java  
import org.springframework.context.annotation.Configuration;  
import org.springframework.context.annotation.Import;  
  
@Configuration  
@Import(GreetingConfig.class) // 导入GreetingConfig配置类  
public class AppConfig {  
    // 其他bean定义可以在这里添加  
}

最后,编写客户端代码来使用这个bean:

// Application.java  
import org.springframework.context.ApplicationContext;  
import org.springframework.context.annotation.AnnotationConfigApplicationContext;  
  
public class Application {  
  
    public static void main(String[] args) {  
        // 创建应用上下文,指定主配置类  
        ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);  
  
        // 从应用上下文中获取GreetingService bean  
        GreetingService greetingService = context.getBean(GreetingService.class);  
  
        // 调用bean的方法并打印结果  
        System.out.println(greetingService.sayGreeting());  
  
        // 关闭应用上下文(虽然在这个简单示例中不是必需的)  
        ((AnnotationConfigApplicationContext) context).close();  
    }  
}

运行上面代码,将会有如下输出:

Hello, World!

和ImportBeanDefinitionRegistrar接口一起使用

ImportBeanDefinitionRegistrar是一个Spring接口,它允许在运行时以编程方式注册额外的bean定义,当需要在Spring容器刷新过程中动态添加bean定义时,可以实现这个接口,ImportBeanDefinitionRegistrar通常与@Import注解结合使用,以便在Spring容器启动时执行自定义的bean注册逻辑。

下面是一个简单的案例,演示了如何使用ImportBeanDefinitionRegistrar来动态注册bean定义。

首先,创建一个简单的服务类MyDynamicService

public class MyDynamicService {  
    public void performTask() {  
        System.out.println("MyDynamicService is performing a task.");  
    }  
}

然后,创建一个实现了 ImportBeanDefinitionRegistrar 接口的类 MyBeanDefinitionRegistrar

import org.springframework.beans.factory.support.BeanDefinitionRegistry;  
import org.springframework.beans.factory.support.GenericBeanDefinition;  
import org.springframework.context.annotation.ImportBeanDefinitionRegistrar;  
import org.springframework.core.type.AnnotationMetadata;  
  
public class MyBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {  
  
    @Override  
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {  
        // 创建 GenericBeanDefinition 实例  
        GenericBeanDefinition beanDefinition = new GenericBeanDefinition();  
        // 设置 bean 类  
        beanDefinition.setBeanClassName(MyDynamicService.class.getName());  
        // 注册 bean 定义到容器中,指定 bean 的名称  
        registry.registerBeanDefinition("myDynamicService", beanDefinition);  
    }  
}

记着,需要一个配置类来触发 MyBeanDefinitionRegistrar 的注册:

import org.springframework.context.annotation.Configuration;  
import org.springframework.context.annotation.Import;  
  
@Configuration  
@Import(MyBeanDefinitionRegistrar.class)  
public class MyAppConfig {  
    // 其他配置...  
}

最后,在应用程序中使用 AnnotationConfigApplicationContext 来加载 MyAppConfig 并获取 MyDynamicService 的实例:

import org.springframework.context.ApplicationContext;  
import org.springframework.context.annotation.AnnotationConfigApplicationContext;  
  
public class Application {  
    public static void main(String[] args) {  
        ApplicationContext context = new AnnotationConfigApplicationContext(MyAppConfig.class);  
        MyDynamicService myDynamicService = context.getBean(MyDynamicService.class);  
        myDynamicService.performTask(); // 输出:"MyDynamicService is performing a task."  
    }  
}

在这个例子中,当Spring容器启动时,它会处理@Import注解并调用MyBeanDefinitionRegistrarregisterBeanDefinitions方法。

这个方法会在容器中动态地注册MyDynamicService的bean定义,随后,可以像获取其他Springbean一样获取并使用MyDynamicService的实例。

和ImportSelector接口一起使用

ImportSelector是Spring框架提供的一个接口,它允许开发者在运行时根据某些条件或逻辑选择要导入的配置类,ImportSelector接口定义了一个方法selectImports,该方法返回一个字符串数组,表示要导入的配置类的全限定名。

以下是一个简单的示例,展示了如何使用ImportSelector来动态选择要导入的配置类:

首先,定义两个简单的配置类ConfigAConfigB,每个配置类都有一个Bean定义:

// ConfigA.java  
@Configuration  
public class ConfigA {  
    @Bean  
    public String configABean() {  
        return "Bean from ConfigA";  
    }  
}  
  
// ConfigB.java  
@Configuration  
public class ConfigB {  
    @Bean  
    public String configBBean() {  
        return "Bean from ConfigB";  
    }  
}

接下来,创建一个实现 ImportSelector 接口的类 MyImportSelector,它根据某个条件(例如系统属性)来决定导入哪个配置类:

// MyImportSelector.java  
import org.springframework.context.annotation.ImportSelector;  
import org.springframework.core.type.AnnotationMetadata;  
  
import java.util.Arrays;  
  
public class MyImportSelector implements ImportSelector {  
  
    @Override  
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {  
        // 根据某个条件决定导入哪个配置类  
        if (System.getProperty("config.selector") != null && "configA".equals(System.getProperty("config.selector"))) {  
            return new String[]{ConfigA.class.getName()};  
        } else {  
            return new String[]{ConfigB.class.getName()};  
        }  
    }  
}

在上面的 MyImportSelector 类中,selectImports 方法检查系统属性 config.selector 的值,并根据该值返回相应的配置类全限定名。

最后,创建一个主配置类 MainConfig,并使用 @Import 注解引入 MyImportSelector

// MainConfig.java  
import org.springframework.context.annotation.Configuration;  
import org.springframework.context.annotation.Import;  
  
@Configuration  
@Import(MyImportSelector.class)  
public class MainConfig {  
    // 其他配置...  
}

现在,在应用程序中,可以根据系统属性 config.selector 的值来动态选择要加载的配置类:

// Application.java  
import org.springframework.context.ApplicationContext;  
import org.springframework.context.annotation.AnnotationConfigApplicationContext;  
  
public class Application {  
    public static void main(String[] args) {  
        // 设置系统属性以决定要导入的配置类  
        System.setProperty("config.selector", "configA"); // 设置为 "configA" 或 "configB"  
  
        ApplicationContext context = new AnnotationConfigApplicationContext(MainConfig.class);  
  
        // 根据系统属性的设置,以下将打印出不同的结果  
        String bean = context.getBean(String.class);  
        System.out.println(bean);  
    }  
}

运行上述 Application 类的 main 方法,并根据需要设置系统属性 config.selector 的值,将看到根据该值动态加载了不同的配置类中的 Bean。如果设置为 “configA”,则输出将是 “Bean from ConfigA”;如果设置为其他值或未设置,则输出将是 “Bean from ConfigB”。

使用@Import注解引入多个类

当使用@Import注解来组合多个配置类时,可以在一个主配置类上使用@Import注解,并指定要导入的其他配置类的类名。这样做可以让Spring框架在创建应用上下文时,加载并处理这些配置类中定义的bean。

以下是一个简单的代码案例,展示了如何使用@Import来组合多个配置类:

首先,定义两个简单的配置类,每个配置类都使用@Bean注解来定义一个不同的bean:

// FirstConfig.java  
@Configuration  
public class FirstConfig {  
  
    @Bean  
    public String firstBean() {  
        return "First Bean";  
    }  
}  
  
// SecondConfig.java  
@Configuration  
public class SecondConfig {  
  
    @Bean  
    public String secondBean() {  
        return "Second Bean";  
    }  
}

记着,创建一个主配置类,并使用@Import注解来导入上面定义的两个配置类:

// MainConfig.java  
@Configuration  
@Import({FirstConfig.class, SecondConfig.class})  
public class MainConfig {  
    // 这里不需要定义任何bean,因为只是组合其他配置类  
}

最后,编写一个客户端类来演示如何从应用上下文中获取这些bean:

// Application.java  
public class Application {  
  
    public static void main(String[] args) {  
        // 创建应用上下文,并指定主配置类  
        ApplicationContext context = new AnnotationConfigApplicationContext(MainConfig.class);  
  
        // 从应用上下文中获取FirstConfig中定义的bean  
        String firstBean = context.getBean("firstBean", String.class);  
        System.out.println("First Bean: " + firstBean);  
  
        // 从应用上下文中获取SecondConfig中定义的bean  
        String secondBean = context.getBean("secondBean", String.class);  
        System.out.println("Second Bean: " + secondBean);  
  
        // 关闭应用上下文(虽然不是必需的,但在实际应用程序中应该这样做)  
        ((ConfigurableApplicationContext) context).close();  
    }  
}

当运行Application类的main方法时,输出将会是:

First Bean: First Bean  
Second Bean: Second Bean

这证明了Spring成功地加载了FirstConfigSecondConfig中定义的bean,并将它们组合到了由MainConfig类创建的应用上下文中,通过这种方式,可以轻松地将多个配置类组合在一起,从而构建更复杂的配置层次结构。

关注我,每天学习互联网编程技术 - 程序员古德

END!
END!
END!

往期回顾

精品文章

Java并发基础:concurrent Flow API全面解析

Java并发基础:CopyOnWriteArraySet全面解析

Java并发基础:ConcurrentSkipListMap全面解析

Java并发基础:ConcurrentSkipListSet全面解析!

Java并发基础:SynchronousQueue全面解析!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/431861.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++面试干货---带你梳理常考的面试题(二)

顾得泉:个人主页 个人专栏:《Linux操作系统》 《C从入门到精通》 《LeedCode刷题》 键盘敲烂,年薪百万! 1.struct 和 class 区别 1.默认访问权限:struct中的成员默认为public,而class中的成员默认为priv…

力扣404 左叶子之和 Java版本

文章目录 题目描述解题思路代码 题目描述 给定二叉树的根节点 root ,返回所有左叶子之和。 示例 1: 输入: root [3,9,20,null,null,15,7] 输出: 24 解释: 在这个二叉树中,有两个左叶子,分别是 9 和 15,所以返回 2…

二手手机管理系统|基于Springboot的二手手机管理系统设计与实现(源码+数据库+文档)

二手手机管理系统目录 目录 基于Springboot的二手手机管理系统设计与实现 一、前言 二、系统设计 三、系统功能设计 1、用户管理功能的实现界面 2、用户中心管理功能的实现界面 3、新闻信息管理功能的实现界面 4、商品收藏管理功能的实现界面 5、订单管理功能的实现界…

2024年3月6日 十二生肖 今日运势

小运播报:2024年3月6日,星期三,农历正月廿六 (甲辰年丁卯月己巳日),法定工作日。 红榜生肖:牛、猴、鸡 需要注意:鼠、虎、猪 喜神方位:东北方 财神方位:正…

【pyinstaller打包记录】Windows系统打包exe后,onnxruntime报警告(Init provider bridge failed)

简介 PyInstaller 是一个用于将 Python 程序打包成可执行文件(可执行程序)的工具。它能够将 Python 代码和其相关的依赖项(包括 Python 解释器、依赖的模块、库文件等)打包成一个独立的可执行文件,方便在不同环境中运行…

【Java设计模式】五、建造者模式

文章目录 1、建造者模式2、案例:共享单车的创建3、其他用途 1、建造者模式 某个对象的构建复杂将复杂的对象的创建 和 属性赋值所分离,使得同样的构建过程可以创建不同的表示建造的过程和细节调用者不需要知道,只需要通过构建者去进行操作 …

【SpringBoot3.x教程 01】SpringBoot简介及工程搭建

前言:什么是SpringBoot? SpringBoot是一个开源的Java基础框架,它被设计来简化Spring应用的初始搭建以及开发过程。这个框架利用了“约定优于配置”的理念,提供了一系列大型项目中常用的默认配置,让开发者可以快速启动和…

kafka查看消息两种方式(命令行和软件)+另附发送消息方式

1、命令行方式 ①找到kafka安装文件夹 ②执行命令 #指定offset为指定时间作为消息起始位置 kafka-consumer-groups.sh \ --bootstrap-server 20.2.246.116:9092 \ --group group_1 \ --topic lanxin_qiao \ --reset-offsets \ --to-datetime 2023-07-19T01:00:00.000 \ -exe…

解决Maven项目中的依赖冲突

1. 排查依赖冲突 在IDEA中下载插件 Maven Helper用于排查依赖版本冲突。 打开项目的pom.xml文件,点击下方的【Dependency Analyzer】按钮切换到依赖解析页面。 2. 解决版本依赖 在依赖解析页面进行依赖冲突排查操作: 点击 【Exclude】 后会在爆红处所对…

Java多线程实现发布和订阅

目录 简介 步骤 1: 定义消息类 步骤 2: 创建发布者 步骤 3: 创建订阅者 步骤 4: 实现发布-订阅模型 前言-与正文无关 生活远不止眼前的苦劳与奔波,它还充满了无数值得我们去体验和珍惜的美好事物。在这个快节奏的世界中,我们往往容易陷入工作的漩涡…

Claude 3正式发布,超越GPT-4,一口气读15万单词,OpenAI最强的大对手!

目录 多模态AI大模型Claude 3(https://www.anthropic.com/news/claude-3-family)Claude 3 的三个版本新增功能,chatgpt没有的使用成本总结 多模态AI大模型Claude 3(https://www.anthropic.com/news/claude-3-family) …

Stable Diffusion 3报告

报告链接:https://stability.ai/news/stable-diffusion-3-research-paper 文章目录 要点表现架构细节通过重新加权改善整流流量Scaling Rectified Flow Transformer Models灵活的文本编码器RF相关论文 要点 发布研究论文,深入探讨Stable Diffuison 3的…

Sora到底有多强?

北京时间2月16日凌晨,OpenAI发布文本生成视频的AI模型Sora,瞬时刷屏科技圈,成为2024年开年“顶流”。 官方称,Sora只需文本就能自动生成高度逼真和高质量的视频,且时长突破1分钟。这是继文本模型ChatGPT和图片模型Dal…

三整数排序问题的解题逻辑

【题目描述】 输入3个整数,从小到大排序后输出。 【样例输入】 20 7 33 【样例输出】 7 20 33 【解析】 本题解法大概有3种: 1、穷举条件法。 此方法先判断a、b、c大小的所有可能,再根据各种可能性输出不同的排序。 思路是先判断a、…

3Dmax最全快捷键大全,赶紧收藏起来练习起来吧

3Dmax做为一款专业的建模软件,有很多快捷键能帮助我们更好地学习,提升自己的能力。 废话不多说,我们一起来看看。 以上就是3dmax最全快捷键大全,看着容易,但是想要掌握好还需要我们多多练习。 本地max跑图太慢的朋友可…

数据链路层----滑动窗口协议的相关计算

目录 1.窗口大小的相关计算 •停等协议: •后退N帧协议: •选择重传协议: 2.信道利用率相关计算 •停等协议的信道利用率: •连续ARQ(后退N帧协议,选择重传协议)的信道利用率:…

SAP PP学习笔记04 - BOM2 -通过Serial来做简单的BOM变式配置,副明细,BOM状态,BOM明细状态,项目种类,递归BOM

本章继续讲BOM。 本章讲通过Serial来做简单的BOM变式配置。还讲了BOM的相关概念:副明细,BOM状态,BOM明细状态,项目种类,递归BOM 等。 1,通过Serial(序列号)来做简单的 VC&#xff0…

软考信息系统项目管理师零基础怎么学习?

软考考信息系统项目管理师,零基础怎么入手高项? 要我说对于没有基础的人群来说零基础考信息系统项目管理师还是有一定的难度的,难就难在需要时间去了解基础,而相对于系统分析师、系统构架设计师、网络规划设计师、系统规划与管理…

C++多态详解

文章目录 多态概念定义及实现构成条件虚函数虚函数的重写override 和 final重载、覆盖、隐藏 抽象类纯虚函数接口继承与实现继承 多态的原理虚函数表原理动态绑定与静态绑定 多继承的虚函数表多继承中的虚函数表 多态 概念 多态是面向对象三大特性中相对复杂的一个&#xff0…

c语言网络编程学习整理 网络编程结构框架 一些常见协议的介绍

1.网络分层:osi体系结构 重点:网络层,传输层。 口诀:物数网传会表应。 可是osi体系过于理想,不过其为原型依旧通用: TCP/IP协议 是Internet事实上的工业标准 2.TCP/IP 4层模型 1)网络接口与…