Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入

文章目录

  • 概述
  • @Autowired注解
    • @Autowired的匹配原则
    • @Autowired的 required 属性
    • 使用@Primary 和@Qualifier消除@Autowired的歧义
      • @Primary 不推荐使用
      • @Qualifier推荐使用
  • 在构造函数/方法中使用@Autowired

概述

Spring Boot2.x-04Spring Boot基础-使用注解装配bean 中讲了如何将Bean装载到IoC容器中,这里我们说下Bean之间的依赖关系,当然了还是基于注解的方式。

xml的方式去描述Bean之间的依赖关系,请参考以前的博客

Spring-bean之间的关系

Spring-基于注解的配置[02自动装载bean]


@Autowired注解

举个例子: Manager可以安排Engineer去根据Engineer的类型做不同的工作

Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第1张图片

接口Engineer的接口方法是coding

package com.artisan.springbootmaster.di.intf;

public interface Engineer {

    void coding();
}

假设有个Java程序猿,实现Engineer接口

package com.artisan.springbootmaster.di.intf.impl;


import com.artisan.springbootmaster.di.intf.Engineer;
import org.springframework.stereotype.Service;

@Service
public class JavaEnginerr implements Engineer {

    @Override
    public void coding() {
        System.out.println("Java Engineer works");
    }
}

我们在实现类JavaEnginerr 上使用@Service注解,使其成为一个受Spring容器管理的bean。


接下来,我们来看下Manager类

package com.artisan.springbootmaster.di;

import com.artisan.springbootmaster.di.intf.Engineer;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

@Component
public class Manager {

    @Autowired
    Engineer engineer;

    public void arrange(){
        engineer.coding();
    }

}

可以通过arrange方法安排engineer工作。 这里Engineer 通过@Autowired让IoC容器自动注入进来。

接着我们使用Java类的方式来初始化IoC容器,通过@Configuration标注其是一个配置类 ,通过ComponetScan来扫描基包下面的标注了注解的类,使其成为受Spring IoC容器托管的bean,方便注入

package com.artisan.springbootmaster.di;

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

@Configuration
@ComponentScan(basePackages = "com.artisan.springbootmaster.*")
public class Config {

}


最后,加载Java类的配置,主要是依靠 AnnotationConfigApplicationContext,启动容器获取bean,并调用对应的方法

package com.artisan.springbootmaster.di;

import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class DITest {

    public static void main(String[] args) {
        ApplicationContext applicationContext = new AnnotationConfigApplicationContext(Config.class);
        Manager manager =  applicationContext.getBean(Manager.class);
        manager.arrange();
    }
}

运行

23:04:08.018 [main] DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory - Returning cached instance of singleton bean 'manager'
Java Engineer works

@Autowired会根据属性的类型( by type )找到对应的 Bean 进行注入。
Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第2张图片

通过结果可以知道,通过注解@Autowired 成功的将JavaEngierr注入到了Manager实例中。


@Autowired的匹配原则

上面这个例子中@Autowired的用法很简单,我们继续来看下@Autowired

当然了,Engineer可能有多个,比如又来了个AndroidEngineer

package com.artisan.springbootmaster.di.intf.impl;

import com.artisan.springbootmaster.di.intf.Engineer;
import org.springframework.stereotype.Service;


@Service
public class AndroidEngineer implements Engineer {

    @Override
    public void coding() {
        System.out.println("Android Engineer works");
    }
}

这是IDEA中可以看到,有提示报错了
Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第3张图片

让我们继续运行下DITest,抛出了异常

 No qualifying bean of type 'com.artisan.springbootmaster.di.intf.Engineer' available: expected single matching bean but found 2: androidEngineer,javaEnginerr

意思很明显,@Autowired根据类型来匹配Engineer,却发现有2个bean都是Engineer类型 ,这下子Spring不知道注入哪个了。

@Autowired的匹配原则:根据类型找到对应的 Bean,如果对应类型的 Bean 不是唯一 的,那么会继续根据其属性名称和 Bean 的名称进行匹配。如果匹配上,就会使用该 Bean,如果还无法匹配,就会抛出异常。

所以根据上面的原则,比较挫的一个办法(这里只是说可以这么改,但是肯定不推荐这么改

既然是两个,那我就让bean的名字一样呗

  • 方法一:Manager中的Engineer engineer保持不变,给这两个Engineer中的任意一个标注@Service(value "engineer"),指定其bean的名字为engineer,这样根据name就匹配上了,同样不会抛出异常。测试通过。
  • 方法二:Manager中的Engineer engineer改为这两个bean的任意一个名字,@Service标注的实现类Bean的名字为默认第一个字母小写其余保持不变,这样name也能匹配上,同样不会抛出异常。测试通过。

Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第4张图片

结果
Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第5张图片

这里只是举例验证下Spring @Autowired的匹配规则,实际工作中并不推荐这么改。。。。


@Autowired的 required 属性

@Autowired 是一个默认必须找到对应 Bean 的注解,如果不能确定其标注属性一定会存在并且允许这个被标注的属性为 null , 那么你可以配置@Autowired 属性 required 为 false.

@Autowired既可以标注在属性上,也可以标注在方法上

 @Autowired(required = false)

使用@Primary 和@Qualifier消除@Autowired的歧义

上面通过修改name,使其name保持一致的方式消除了歧义,可以正常的注入,不过并不推荐。

@Primary 不推荐使用

也可以使用@Primary,当然了,也不推荐这么干。 因为另外一个类也可以标注@Primary,Spring又无法知道注入哪个了。

注解@Primary是修改优先权的注解,像上面的两个例子,有2个beanandroidEngineer,javaEnginerr, 如果我们仅在JavaEnginerr这个类上标注@Primary,意思是告诉Spring IoC 容器 , 当发现有多个同样类型的 Bean ,请优先使用标注了@Primary的这个bean进行注入。

Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第6张图片

结果:
Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第7张图片


@Qualifier推荐使用

@Qualifier的value属性定义bean的名,该名称将会和@Autowired 组合在一起,通过类型和名称一起找到 Bean。Spring IoC容器中Bean 名称是唯一的标识,通过这个就可以消除歧义了

Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第8张图片

结果

Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第9张图片

即使 JavaEnginerr标注了@Primary,但是由于使用了@Qualifier,注入的依然是androidEngineer.


在构造函数/方法中使用@Autowired

上面的例子,我们是在属性上使用的@Autowired.
如果使构造函数呢?

我们改造下Manager

package com.artisan.springbootmaster.di;

import com.artisan.springbootmaster.di.intf.Engineer;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Component;

@Component
public class Manager {

    Engineer engineer;

    public void arrange(){
        engineer.coding();
    }


    public Manager(@Autowired @Qualifier("javaEnginerr") Engineer engineer){
        this.engineer = engineer;
    }
}

  • 取消标注在属性上的注解
  • 在构造函数上增加注解,使用方法一样。
  • 如果仅有一个类型的Bean, @Qualifier就没有必要加上了。

运行

Spring Boot2.x-05Spring Boot基础-使用注解完成依赖注入_第10张图片

你可能感兴趣的:(【Spring,Boot2.X】)