认识Spring中的@Import注解

简介

在Spring 3.0,Spring引入了@Import注解,允许通过JavaConfig导入一个或多个类作为 Spring Bean,这些类不需要标注 Spring 模式注解。

Spring 模式注解:
@Componet、@Service、@Controller、@Repository、@Configuration

@Import的源码

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Import {

	/**
	 * {@link Configuration @Configuration}, {@link ImportSelector},
	 * {@link ImportBeanDefinitionRegistrar}, or regular component classes to import.
	 */
	Class<?>[] value();

}
  • 可以看出@Import可以配合 Configuration , ImportSelector, ImportBeanDefinitionRegistrar 来使用,也可以导入普通的类。
  • @Import只允许放到类上面,不能放到方法上。

准备

搭建Spring环境,创建启动类``:

public class MyApplication {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext("com.haan");
        String[] beanDefinitionNames = context.getBeanDefinitionNames();
        Stream.of(beanDefinitionNames)
                .forEach(s -> System.out.println(s));
    }
}

导入普通的类

新建两个类,TestService1.class 和TestService2.class:

public class TestService1 {

    public void print(){
        System.out.println("this is service01 !");
    }
}

public class TestService2 {

    public void print(){
        System.out.println("this is service02 !");
    }
}

定义 MyImportConfiguration类,注解@Configuration,@Import,@Import 传参values={Test}value = {TestService1.class,TestService2.class}:

@Import(value = {TestService1.class,TestService2.class})
@Configuration
public class MyImportConfiguration {

}

执行main函数,输出日志如下:

org.springframework.context.annotation.internalConfigurationAnnotationProcessor
org.springframework.context.annotation.internalAutowiredAnnotationProcessor
org.springframework.context.annotation.internalCommonAnnotationProcessor
org.springframework.context.event.internalEventListenerProcessor
org.springframework.context.event.internalEventListenerFactory
myImportConfiguration
com.haan.TestService1
com.haan.TestService2

Process finished with exit code 0

我们看到,TestService1.class和TestService2.class 都被实例为Spring Bean 了。但是这种方式有一些问题,那就是只能使用类的无参构造方法来创建bean,对于有参数的构造方法就无能为力了。

结合ImportSelector接口

我们先看一下ImportSelector接口:

public interface ImportSelector {

    // Select and return the names of which class(es) should be imported based on
	String[] selectImports(AnnotationMetadata importingClassMetadata);

	// Return a predicate for excluding classes from the import candidates
	@Nullable
	default Predicate<String> getExclusionFilter() {
		return null;
	}

}

我们注释可以大概了解到方法selectImport(...)返回的classNames 将会被import,实例为Spring Bean。方法getExclusionFilter()将返回一个Predicate, 后续将通过该Predicate 将候选的类排除,不进行import。

我们这里定义类MyImportSellectorImpl实现接口ImportSelector:

public class MyImportSelectorImpl implements ImportSelector {
    @Override
    public String[] selectImports(AnnotationMetadata annotationMetadata) {
        return new String[]{TestService3.class.getName(),TestService4.class.getName()};
    }
}

我们这里返回了类TestService3TestService4的类名称,TestService3TestService4是我们新创建的两个类:

public class TestService3 {
    public void print(){
        System.out.println("this is service03 !");
    }
}
public class TestService4 {
    public void print(){
        System.out.println("this is service04 !");
    }
}

然后,我们修改下我们的MyImportConfiguration类,将MyImportSelectorImpl添加到@Import的参数中:

@Import(value = {TestService1.class,TestService2.class,MyImportSelectorImpl.class})
@Configuration
public class MyImportConfiguration {

}

再次执性程序,得到控制台输出:

org.springframework.context.annotation.internalConfigurationAnnotationProcessor
org.springframework.context.annotation.internalAutowiredAnnotationProcessor
org.springframework.context.annotation.internalCommonAnnotationProcessor
org.springframework.context.event.internalEventListenerProcessor
org.springframework.context.event.internalEventListenerFactory
myImportConfiguration
com.haan.TestService1	// 普通类导入
com.haan.TestService2   // 普通类导入
com.haan.TestService3	// 结合ImportSelector 接口
com.haan.TestService4   // 结合ImportSelector 接口

Process finished with exit code 0

通过输出,我们发现类TestService3 和 TestService4也被实例为 Spring Bean 了。

结合ImportBeanDefinitionRegistrar接口

ImportBeanDefinitionRegistrar接口的源码如下:

public interface ImportBeanDefinitionRegistrar {

	//  Register bean definitions as necessary based on the given annotation metadata of
	default void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry,
			BeanNameGenerator importBeanNameGenerator) {

		registerBeanDefinitions(importingClassMetadata, registry);
	}
   //  Register bean definitions as necessary based on the given annotation metadata of
	default void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
	}
}
  • AnnotationMetadata:通过这个参数可以拿到类的元数据信息
  • BeanDefinitionRegistry:通过这个参数可以操作IOC容器
  • BeanNameGenerator : 通过这个可以操作Bean 名称生成策略

我们可以使用一个类来实现这个接口:

public class MyImportBeanDefinitionRegistrarImpl implements ImportBeanDefinitionRegistrar {
    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
        BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(TestService5.class);
        registry.registerBeanDefinition("testService060001",beanDefinitionBuilder.getBeanDefinition());
    }
}

其中的TestService5 是我们新创建的一个类:

public class TestService5 {
    public void print(){
        System.out.println("this is service05 !");
    }
}

然后,我们修改下我们的MyImportConfiguration类,将MyImportBeanDefinitionRegistrarImpl添加到@Import的参数中:

@Import(value = {TestService1.class,TestService2.class,MyImportSelectorImpl.class,MyImportBeanDefinitionRegistrarImpl.class})
@Configuration
public class MyImportConfiguration {

}

再次执性程序,得到控制台输出:

org.springframework.context.annotation.internalConfigurationAnnotationProcessor
org.springframework.context.annotation.internalAutowiredAnnotationProcessor
org.springframework.context.annotation.internalCommonAnnotationProcessor
org.springframework.context.event.internalEventListenerProcessor
org.springframework.context.event.internalEventListenerFactory
myImportConfiguration
com.haan.TestService1
com.haan.TestService2
com.haan.TestService3
com.haan.TestService4
testService060001			// 结合ImportBeanDefinitionRegistrar 

Process finished with exit code 0

我们发现类TestService5 也被实例为 Spring Bean 了, 且 Bean Name 可以自定义。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: @Import注解Spring框架的一种注解,可以用来快速导入多个组件,包括类、包或者配置类。 使用@Import注解可以在一个配置类快速导入其他组件,而不必通过@Bean或@ComponentScan注解来导入。这样可以方便地将功能模块化,并且使用起来也更加方便。 使用方法如下: 1. 在配置类上使用@Import注解,并指定要导入的组件的类型数组。 例如: ``` @Import({MyConfiguration.class, MyBean.class}) public class AppConfig { // ... } ``` 2. 也可以使用ImportSelector接口和ImportBeanDefinitionRegistrar接口来动态选择和注册组件。 例如: ``` @Import(MyImportSelector.class) public class AppConfig { // ... } public class MyImportSelector implements ImportSelector { @Override public String[] selectImports(AnnotationMetadata importingClassMetadata) { // 在这里可以动态的返回需要导入的组件 return new String[] {MyBean.class.getName()}; } } ``` 使用@Import注解导入组件的好处是可以方便地将组件模块化,可以将一个功能模块分成若干个配置类,然后使用@Import注解导入。这样可以让代码更加清晰,更加方便维护。 ### 回答2: @Import注解Spring框架的一种注解,在使用该注解时,可以将其他配置类或者Bean定义类导入到当前配置类。通过@Import注解,可以在一个配置类同时引入多个不同的配置类或者Bean定义类,实现了配置的模块化和复用。 @Import注解可以实现多种功能,具体包括以下几点: 1. 导入配置类:通过@Import注解可以将其他的配置类导入到当前的配置类,这样可以将多个配置类组合在一起,实现配置的分离和复用,提高代码的可维护性和可读性。 2. 导入Bean定义类:除了配置类之外,@Import注解也可以将其他的Bean定义类引入到当前配置类,这样可以将多个不同的Bean定义类组合在一起,实现Bean的组装和扩展。 3. 导入自动配置类:Spring Boot框架的自动配置就是通过@Import注解来实现的,通过将相应的自动配置类导入到配置类,可以实现对应功能的自动配置和初始化,减少了开发人员的工作量。 4. 导入条件配置:通过@Import注解可以根据不同的条件来选择性地导入不同的配置类或者Bean定义类,根据具体的条件来进行动态的选择和配置,实现更加灵活和可配置化的开发。 总结来说,@Import注解Spring框架起到了组合和扩展配置的作用,可以将多个配置类或者Bean定义类导入到当前配置类,实现了配置的模块化和复用,同时也提供了条件导入的功能,使得配置的选择更加灵活和可配置化。 ### 回答3: @Import注解Spring框架的一个注解,作用是用于导入其他的配置类或者Bean。通过@Import注解,我们可以将其他的配置类或者Bean引入到当前的配置类,从而实现配置类之间的解耦。 @Import注解可以用于导入其他的@Configuration配置类,这样可以方便地将多个配置类合并到一个总的配置类。通过@Import注解,我们可以将不同的配置类按照逻辑进行划分,分别编写配置类,然后使用@Import注解将它们引入到一个总的配置类。这样做的好处是,能够更好地组织和管理配置类,提高代码的可读性和可维护性。 除了导入其他的配置类,@Import注解还可以用于导入其他的普通的Bean。这对于一些无法通过@Configuration注解进行配置的Bean来说非常有用。通过@Import注解,我们可以将这些Bean引入到当前的配置类,然后使用@Autowired注解进行注入。 总之,@Import注解的作用是使得配置类之间可以进行解耦,并且可以方便地引入其他的配置类或者Bean。它在Spring框架的应用非常广泛,能够有效地提高代码的可读性、可维护性和复用性。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值