Spring Import注解

今天了解了,Spring @Import的使用

先贴上Spring官方关于Spring @Import注解的文档链接   https://docs.spring.io/spring/docs/3.0.x/spring-framework-reference/html/beans.html#beans-java-using-import

 

一.@Import 引入一个普通java对象 适用4.2.0之后版本

二.@Import与@Configuration使用方式

三.@Import 与 ImportBeanDefinitionRegistrar 使用4.2.0之前版本

四.@Import 与 ImportSelector 使用方式

 

 

 

一。@Import直接引入一个类,将其作为Spring bean,受Spring容器管理;(Spring 4.2.0以及之后版本可以使用)

在Spring 4.2.x之前的版本,@Import注解无法引入一个没有注解 @Configuration 或者 @Bean 且没有实现ImportBeanDefinitionRegistry 或者 ImportSelector 的类,将其作为Spring Bean管理;

  如果你在使用的版本是4.2.0之前的版本,想要使用@Import导入一个类,方法二、三或许可行;

特地拿了Spring 4.1.9版本,测试,@Import一个普通的Java类,抛出异常:

Exception in thread "main" org.springframework.beans.factory.parsing.BeanDefinitionParsingException: Configuration problem: pack1.AppConfig2 was @Import'ed but is not annotated with @Configuration nor does it declare any @Bean methods; it does not implement ImportSelector or extend ImportBeanDefinitionRegistrar. Update the class to meet one of these requirements or do not attempt to @Import it.

翻译过来就是: AppConfig2被import导入了,但是没有标注@Configuration或者也没有声明任何@Bean方法,也没有实现ImportSelector或者继承ImportBeanDefinitionRegistrar类;

 

不做任何改动,修改maven导入的Spring版本号,拿4.2.0测试,就成功Import该java类;  Import导入的类,默认beanId是类全限定名,可以设定@Configuration改变类的beanId;

关于为什么配置类上加@Configuration作用,查看我的理解      https://www.cnblogs.com/lvbinbin2yujie/p/10279416.html

@Configuration
@Import({AppConfig2.class})
public class AppConfig1 {

    public static void main(String[] args) {
        AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(AppConfig1.class);
        String[] names = ac.getBeanDefinitionNames();
        for (String string : names) {
            System.out.println(string+","+ac.getBean(string));
        }
    }
}


public class AppConfig2 {
    
}

 

二。@Import引入配置类@Configuration标注的类; 纯属个人看Spring官方文档的理解,大致能看懂,Spring官方文档链接:https://docs.spring.io/spring/docs/3.0.x/spring-framework-reference/html/beans.html#beans-java-using-import

   2.1 先说最容易看懂的一种使用方式,每个@Configuration里的@Bean定义都很明显,就像开发过程中写了好多配置文件,然后在某一个XML里面<import  />其他的bean;

@Configuration
public class ConfigA {

    @Bean
    public A a() {
        return new A();
    }
}

public class A {

}

 

 

@Import({ConfigA.class})
@Configuration
public class ConfigB {

    @Bean
    public B b() {
        return new B();
    }
    
    public static void main(String[] args) {
        AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(ConfigB.class);
        String[] names = ac.getBeanDefinitionNames();
        for (String string : names) {
            System.out.println(string+" , "+ac.getBean(string));
        }
    }
}

 

测试结果如下:  印证了 1. @Import引入的类默认beanId为类全路径名  ;  2. @Configuration标注的类是动态代理之后的类 ;

configB , pack1.ConfigB$$EnhancerBySpringCGLIB$$1acf4932@4313f5bc
pack1.ConfigA , pack1.ConfigA$$EnhancerBySpringCGLIB$$e26ca611@7f010382
a , pack1.A@1e802ef9
b , pack1.B@2b6faea6

 

 

  2.2 但是我们平常见得开发中,比如service配置文件在一个XML里,dao的配置文件在一个XML中,就会跨文件引用,但是我们都会写 ref引用那个对象,因为Spring会帮助我们找到需要的对象;

举个例子模拟这种跨文件的引用,Spring官方文档里面也有这样一个例子;

// dao类
public class AccountDao {

    private DataSource dataSource;
    
    public AccountDao(DataSource dataSource) {
        this.dataSource=dataSource;
    }
    
    public void saveMoney() {
        System.out.println("利用dataSource模拟存钱操作");
    }

}

 

// service类
public class AccountService {

    private AccountDao accountDao;
    
    public AccountService(AccountDao accountDao) {
        this.accountDao=accountDao;
    }
    
    public void saveMoney() {
        accountDao.saveMoney();
    }
}

 

下面将使用注解,来代替原来的XML配置方式,定义了两个配置类DaoConfig  ServiceConfig 

@Configuration
public class DaoConfig {

    @Autowired
    private DataSource dataSource;
    
    @Bean
    public AccountDao accountDao() {
        return new AccountDao(dataSource);
    }
}

 

@Configuration
public class ServiceConfig {

    @Autowired
    private AccountDao accountDao;
    
    @Bean
    public AccountService accountService() {
        return new AccountService(accountDao);
    }
    

 

@Configuration
@Import({ServiceConfig.class,DaoConfig.class})
public class SystemConfig {
    
    @Bean
    public DataSource dataSource() {
        return new DruidDataSource();
    }
    
    public static void main(String[] args) {
        AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(SystemConfig.class);
        AccountService service = ac.getBean(AccountService.class);
        service.saveMoney();   
    }
}

 

 

查看测试结果:

利用dataSource模拟存钱操作

 

 

简单来讲下我的分析:从bean的load讲一下, 第一个加载的bean是SystemConfig,先解析Import注解,和SysemConfig一样递归加载ServiceConfig以及DaoConfig这两个类,递归因为没有别的注解了,所以就是解析@Bean,也就是纳入了AccountService AccountDao 最后是 DataSource 这三个Bean ,别忘了 加上SystemConfig、ServiceConfig 、DaoConfig这样六个bean对象;然后属性@Autowired注入,如果属性注入的时候这个bean还没有就回去实例化这个Bean,这样一来就完成了。

可能看完这个方法,还是复杂啊,假如少Import一个配置类,那肯定会报错了。

 

  2.3 此外,Spring另外一个方法可以实现上面功能,比较奇特的方式;

@Configuration
public class DaoConfig {

    @Autowired
    private DataSource dataSource;
    
    @Bean
    public AccountDao accountDao() {
        return new AccountDao(dataSource);
    }
}

 

@Configuration
public class ServiceConfig2 {

    @Autowired
    private DaoConfig daoConfig;
    
    @Bean
    public AccountService accountService() { 
        AccountDao dao1 = daoConfig.accountDao();
        System.out.println("ServiceConfig中的dao:"+dao1);
        return new AccountService(dao1);
    }
}

 

@Configuration
@Import({ServiceConfig2.class,DaoConfig.class})
public class SystemConfig2 {

    @Autowired
    private ServiceConfig2 serviceConfig;
    
    @Bean
    public DataSource dataSource() {
        return new DruidDataSource();
    }
    
    public static void main(String[] args) {
        AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(SystemConfig2.class);
        AccountService service = ac.getBean(AccountService.class);
        service.saveMoney();
    }
}

 

 

测试结果: 由于使用main方法测试,如果在业务方法中可以直接 serviceConfig.accountService().saveMoney();

ServiceConfig中的dao:pack2.AccountDao@3daa422a
利用dataSource模拟存钱操作

 

 

说明: 1. @Configuration注解的类本身也是个bean对象,这点不能忘;   2.@Configuration标注的类调用 同一个@Bean方法 会一直得到同一个bean对象 ;

    3. Spring官方文档中有一句:要使用@Configuration注解需要使用 <context:component-scan /> 标签

 

三。方法三,实现ImportBeanDefinitionRegistrar接口, 这也是 Spring 4.2.0之前版本 如果想要引入一个普通bean,没法直接Import,实现接口并且重写方法里注册这个bean对象;

public class ABeanDefinitionRegistry implements ImportBeanDefinitionRegistrar{

    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata,
            BeanDefinitionRegistry registry) {
     //简单说下使用,只要将beanDefinition注册到registry里,就算把bean对象引入了;只是简单写了个例子引入A类 BeanDefinitionBuilder bd
= BeanDefinitionBuilder.rootBeanDefinition(A.class); registry.registerBeanDefinition("a", bd.getBeanDefinition()); } }

 

@Import({ABeanDefinitionRegistry.class})

 

这个方法是怎么调用的呢?ConfigurationClassParser类的processImports方法,是解析类上@Import注解的;ConfigurationClassBeanDefinitionReader的loadBeanDefinitionsFromRegistrars方法,遍历引入的实现ImportBeanDefinitionRegistrar的类,然后调用其registerBeanDefinitions方法; 

 

四。方法四,实现ImportSelector接口,这种方式相比之前方式更加容易理解,只需要返回bean的类全限定名即可;

public class AImportSelector implements ImportSelector{
    @Override
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        return new String[] {A.class.getName()};
    }
}

 

这样调用即可;

@Import({AImportSelector.class})

 

 三、四两种方式都有个参数为AnnotationMetadata是类上的注解信息的,可以这样实现很多功能,由于不是很了解;

 

posted @ 2019-01-18 20:17  喜欢日向雏田一样的女子啊  阅读(2218)  评论(0编辑  收藏  举报