Spring:了解@Import注解的三种用法

一、前言

  在 Spring 框架中,@Import 注解用于导入配置类,使得你可以在一个配置类中引入另一个或多个配置类,从而实现配置的模块化。这对于组织大型应用程序的配置非常有用,因为它允许你将配置分散到多个类中,然后再将它们组合在一起。

  本文将详细介绍@Import注解的三种方式,并通过示例演示它们的用法。通过对本文的阅读,读者将更好地理解@Import注解在Spring框架中的作用和应用。

二、 @Import 的三种用法

1. 导入普通的配置类

  当你有多个配置类,并且想要将它们组合成一个更大的配置时,可以使用 @Import 来导入其他配置类。

代码示例:

// 配置类 A  
@Configuration  
public class MyConfigA {  
    @Bean  
    public MyServiceA myServiceA() {  
        return new MyServiceA();  
    }  
}  
// 配置类 B,它导入了配置类 A  
@Configuration  
@Import(MyConfigA.class)  
public class MyConfigB {  
  
    @Bean  
    public MyServiceB myServiceB(MyServiceA myServiceA) {  
        return new MyServiceB(myServiceA);  
    }  
}  
// 服务类 A  
public class MyServiceA {  
    // 业务处理 
}  

// 服务类 B  
public class MyServiceB {  
  
    private final MyServiceA myServiceA;  
  
    public MyServiceB(MyServiceA myServiceA) {  
        this.myServiceA = myServiceA;  
    }  
  
    // 代码逻辑 
}

  在上面例子中,MyConfigB 通过 @Import 导入了 MyConfigA,从而能够在 MyServiceB 中注入 MyService。

2. 导入实现了 ImportSelector 接口的类

  当你想根据条件动态地导入配置类时,可以实现 ImportSelector 接口。

代码示例:

// ImportSelector 实现类  
public class MyImportSelector implements ImportSelector {  
  
    @Override  
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {  
        // 这里简单起见,始终返回 MyConfigA.class 的名称  
        return new String[]{MyConfigA.class.getName()};  
    }  
}  
// 配置类 B,它导入了 MyImportSelector  
@Configuration  
@Import(MyImportSelector.class)  
public class MyConfigB {  
    // 代码逻辑  
} 

  在这个例子中,MyConfigB 通过 @Import 导入了 MyImportSelector,而 MyImportSelector 会根据条件选择性地返回要导入的配置类名称。在这个简单的例子中,它始终返回 MyConfigA.class 的名称。

3. 导入实现了 ImportBeanDefinitionRegistrar 接口的类

  当你想在导入配置时直接注册 BeanDefinition 到 Spring 容器中时,可以实现 ImportBeanDefinitionRegistrar 接口。

代码示例:

// ImportBeanDefinitionRegistrar 实现类  
public class MyBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {  
  
    @Override  
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {  
        RootBeanDefinition beanDefinition = new RootBeanDefinition(MyServiceA.class);  
        registry.registerBeanDefinition("myServiceA", beanDefinition);  
    }  
}  
// 配置类 B,它导入了 MyBeanDefinitionRegistrar  
@Configuration  
@Import(MyBeanDefinitionRegistrar.class)  
public class MyConfigB {  
  
    @Bean  
    public MyServiceB myServiceB(MyServiceA myServiceA) {  
        return new MyServiceB(myServiceA);  
    }  
}  

  在这个例子中,MyConfigB 通过 @Import 导入了 MyBeanDefinitionRegistrar,而 MyBeanDefinitionRegistrar 在 registerBeanDefinitions 方法中直接注册了一个 MyServiceA 的 BeanDefinition 到 Spring 容器中。这样,即使没有 MyConfigA 类,MyServiceB 也可以依赖注入 MyServiceA

三、 项目实战

1. 准备一个实体类

package com.example.yddemo.test;

public class User {

    private String name;

    private Integer age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

2. 定义TestDefinitionRegistrar

package com.example.yddemo.test;

import org.springframework.beans.factory.support.BeanDefinitionRegistry;
import org.springframework.beans.factory.support.RootBeanDefinition;
import org.springframework.context.annotation.ImportBeanDefinitionRegistrar;
import org.springframework.core.type.AnnotationMetadata;

public class TestDefinitionRegistrar implements ImportBeanDefinitionRegistrar {

    @Override
    public void registerBeanDefinitions(AnnotationMetadata annotationMetadata,
                                        BeanDefinitionRegistry beanDefinitionRegistry) {
        boolean beanDefinition = beanDefinitionRegistry.containsBeanDefinition("com.example.yddemo.test.User");
        if(beanDefinition){
            RootBeanDefinition rootBeanDefinition = new RootBeanDefinition(User.class);
            beanDefinitionRegistry.registerBeanDefinition("user",rootBeanDefinition);

        }
    }

}

3. 定义MySelector 选择器

package com.example.yddemo.test;

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

public class MySelector {

    public class SuperSelector implements ImportSelector {
        @Override
        public String[] selectImports(AnnotationMetadata annotationMetadata) {

            return new String[]{
                    "com.example.yddemo.test.User"
            };
        }
    }


}

4. 定义配置类

package com.example.yddemo.test;

import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import;

@Configuration
@Import({User.class, MySelector.class,TestDefinitionRegistrar.class})
public class TestConfig {
}

5. 运行输出结果

package com.example.yddemo.test;

import org.junit.jupiter.api.Test;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class TestDemo {
    AnnotationConfigApplicationContext applicationContext =
            new AnnotationConfigApplicationContext(TestConfig.class);

    @Test
    public void testImport(){
        printBeans(applicationContext);
        User user = applicationContext.getBean(User.class);
        System.out.println("==================" + user);
    }


    private void printBeans(AnnotationConfigApplicationContext applicationContext){
        String[] definitionNames = applicationContext.getBeanDefinitionNames();
        for (String name : definitionNames) {
            System.out.println(name);
        }
    }
}

在这里插入图片描述

总结:
  在本文中,我们介绍了Spring框架中的@Import注解及其三种方式。通过使用@Import注解,开发人员可以灵活地导入不同的配置类、依赖项以及自定义配置,从而更好地组织和管理应用程序的配置。通过示例演示,我们展示了@Import注解在不同场景下的用法,并强调了它在提高代码可维护性和可读性方面的重要性。通过阅读本文,读者可以更好地理解@Import注解在Spring框架中的作用和应用。

@Import注解Spring中的注解之一,用于导入其他的配置类或者普通的类,从而使它们成为当前配置类中的一个Bean。可以用在@Configuration注解的类中,或者用在普通的类中作为一个注解。 使用方法: 1. 导入配置类:可以将一个或多个@Configuration注解的配置类导入到当前配置类中,以便使用其中的Bean。示例代码如下: ```java @Configuration @Import({Config1.class, Config2.class}) public class AppConfig { // ... } ``` 2. 导入普通类:可以将一个或多个非@Configuration注解的普通类导入到当前配置类中,以便使用其中的Bean。示例代码如下: ```java @Configuration @Import({Service1.class, Service2.class}) public class AppConfig { // ... } ``` 3. 导入ImportSelector实现类:可以将一个实现了ImportSelector接口的类导入到当前配置类中,该类可以动态地选择要导入的类。示例代码如下: ```java @Configuration @Import(MyImportSelector.class) public class AppConfig { // ... } ``` 4. 导入ImportBeanDefinitionRegistrar实现类:可以将一个实现了ImportBeanDefinitionRegistrar接口的类导入到当前配置类中,该类可以动态地注册Bean定义。示例代码如下: ```java @Configuration @Import(MyImportBeanDefinitionRegistrar.class) public class AppConfig { // ... } ``` 5. 导入FactoryBean:可以将一个实现了FactoryBean接口的类导入到当前配置类中,以便使用其中的Bean。示例代码如下: ```java @Configuration @Import(MyFactoryBean.class) public class AppConfig { // ... } ``` 需要注意的是,@Import注解只是将其他的类导入到当前配置类中,而并不会自动将其实例化为Bean。如果需要使用导入的类中的Bean,需要在当前配置类中通过@Bean注解手动创建相应的Bean。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值