所谓spring自动注入,是指容器中的一个组件中需要用到另一个组件(例如聚合关系)时,依靠spring容器创建对象,而不是手动创建,主要有三种方式:
1. @Autowired注解——由spring提供
2. @Resource注解——由JSR-250提供
3. @Inject注解——由JSR-330提供
 
@Autowired注解的使用方法
@Target({ElementType.CONSTRUCTOR, ElementType.METHOD, ElementType.PARAMETER, ElementType.FIELD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Autowired { /**
* Declares whether the annotated dependency is required.
* <p>Defaults to {@code true}.
*/
boolean required() default true; }
从源码可以看出:该注解可以用在构造器、方法、参数、属性上,最常见的是用在属性上。
该注解只有一个属性: required,默认为true,如果找不到匹配的bean则报错;设置为false,如果找不到匹配的bean则注入null,并不会报错。
可以配合@Qualifier使用,用于精准指定要注入的bean的名称。
可以配合@Primary使用,当容器中存在多个相同类型的组件时,用于指定优先加载哪一个,这个注解不能用在2个或更多同类型的组件上。
 
1. 当容器中只有一个该类型的组件时
|-不使用@Qualifier,会按bean类型查找,即applicationContext.getBean(bean.class),然后注入这个唯一的bean。
|-使用@Qualifier,会按Qualifier的value值跟bean名称匹配查找,即applicationContext.getBean("Qualifier的value值")。
2. 当容器中没有该类型的组件时
|-required=true——报错expected at least 1 bean which qualifies as autowire candidate。
|-required=false——注入null。
3. 当容器中存在多个该类型的组件时
|-不使用@Qualifier和@Primary时,会按属性名跟bean名称匹配查找,即applicationContext.getBean("属性名")。
|-使用@Primary、但不使用@Qualifier时,会优先加载带有@Primary注解的组件。
|-使用@Qualifier时,不管有没有使用@Primary,都会直接按Qualifier的value值跟bean名称匹配查找。
综上,当容器中存在多个同类型的组件时,加载优先级:@Qulifier>@Primary>属性名,例如下面这个容器中包含3个BookDao组件
@Configuration
@ComponentScan(basePackages = {"cn.monolog.service"})
public class AutowiredBeanConfig { @Bean(value = "bookDao1")
public BookDao bookDao1() {
BookDao bookDao = new BookDao();
bookDao.setLabel("bookDao1");
return bookDao;
} @Bean(value = "bookDao2")
@Primary
public BookDao bookDao2() {
BookDao bookDao = new BookDao();
bookDao.setLabel("bookDao2");
return bookDao;
} @Bean(value = "bookDao3")
public BookDao bookDao3() {
BookDao bookDao = new BookDao();
bookDao.setLabel("bookDao3");
return bookDao;
}
}
自动注入方式是这样的:
@Autowired
@Qualifier(value = "bookDao1")
private BookDao bookDao3;
按照优先级顺序,@Qulifier(bookDao1) > @Primary(bookDao2) > 属性名(bookDao3),最终加载的是名称为bookDao1的组件。
 

@Resource注解的使用方法
@Resource注解的使用跟@Autowired注解类似,但是需要注意:
1. 不支持@Primary注解,也不支持reuqired=false,即不允许注入null;
2. 该注解有一个属性name,类似于@Qualified精准匹配,优先级最高;
3. 默认按照属性名跟bean的名称匹配查找,如果不存在,再按类型匹配查找。
例如,下面这个容器中有两个CarDao组件

@Configuration
@ComponentScan(basePackages = {"cn.monolog.service"})
public class ResourceBeanConfig { @Bean(value = "carDao1")
public CarDao carDao1() {
CarDao carDao = new CarDao();
carDao.setLabel("1");
return carDao;
} @Bean(value = "carDao2")
public CarDao carDao() {
CarDao carDao = new CarDao();
carDao.setLabel("2");
return carDao;
}
}
自动注入方式如下,会按属性名注入carDao2组件。
//自动注入
@Resource
private CarDao carDao2;
但是改为下面这种注入方式:
//自动注入
@Resource(name = "carDao1")
private CarDao carDao2;
由于使用了name精准匹配,会忽略属性名,注入carDao1组件。
 

@Inject注解的使用方法
@Inject注解的使用方法跟@Autowired也基本相似,但是需要注意
1. 使用前需要导入jar包——javax.inject;
2. 支持@Primary注解,而且因为没有精确匹配,@Primary的优先级最高;
2. 不支持required=false,即不能注入null,如果找不到组件肯定报错;
3. 默认按照属性名跟bean的名称匹配查找,如果不存在,再按类型匹配查找。
 
例如,下面这容器中有一个EmployeeDao组件

@Configuration
public class InjectBeanConfig {
@Bean(value = "employeeDao1")
public EmployeeDao employeeDao1() {
EmployeeDao employeeDao = new EmployeeDao();
employeeDao.setLabel("1");
return employeeDao;
}
}
自动注入的方式如下
@Inject
private EmployeeDao employeeDao3;
spring会先按属性名查找名称为employeDao3的组件,即applicationContext.getBean("employeeDao3"),结果不存在;
然后按照类型查找,即applicationContext.getBean(EmployeDao.class),找到employeDao1组件,成功注入。
 
如果容器中有多个同类型组件,例如
@Configuration
public class InjectBeanConfig {
@Bean(value = "employeeDao1")
public EmployeeDao employeeDao1() {
EmployeeDao employeeDao = new EmployeeDao();
employeeDao.setLabel("1");
return employeeDao;
}
@Bean(value = "employeeDao2")
public EmployeeDao employeeDao2() {
EmployeeDao employeeDao = new EmployeeDao();
employeeDao.setLabel("2");
return employeeDao;
}
@Bean(value = "employeeDao3")
public EmployeeDao employeeDao3() {
EmployeeDao employeeDao = new EmployeeDao();
employeeDao.setLabel("3");
return employeeDao;
}
}
注入方式还是这样
@Inject
private EmployeeDao employeeDao3;
仍然会按属性名和bean的名称匹配,即applicationContext.getBean("employeeDao3"),找到employee3,成功注入。
 
但是如果其中某个组件加了@Primary注解,会忽略属性名,优先注入,例如
@Configuration
public class InjectBeanConfig {
@Bean(value = "employeeDao1")
public EmployeeDao employeeDao1() {
EmployeeDao employeeDao = new EmployeeDao();
employeeDao.setLabel("1");
return employeeDao;
}
@Bean(value = "employeeDao2")
@Primary
public EmployeeDao employeeDao2() {
EmployeeDao employeeDao = new EmployeeDao();
employeeDao.setLabel("2");
return employeeDao;
}
@Bean(value = "employeeDao3")
public EmployeeDao employeeDao3() {
EmployeeDao employeeDao = new EmployeeDao();
employeeDao.setLabel("3");
return employeeDao;
}
}
无论注入时使用什么样的属性名,都会注入employeeDao2。

spring自动注入的三种方式的更多相关文章

  1. Spring静态注入的三种方式

    版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/chen1403876161/article/details/53644024Spring静态注入的三 ...

  2. SSH深度历险记(八) 剖析SSH核心原则+Spring依赖注入的三种方式

           于java发育.一类程序猿必须依靠类的其他方法,它是通常new依赖类的方法,然后调用类的实例,这样的发展问题new良好的班统一管理的例子.spring提出了依赖注入的思想,即依赖类不由程 ...

  3. SSH深度历险(八) 剖析SSH核心原理+Spring依赖注入的三种方式

           在java开发中,程序员在某个类中需要依赖其它类的方法,则通常是new一个依赖类再调用类实例的方法,这种开发存在的问题是new的类实例不好统一管理,spring提出了依赖注入的思想,即依 ...

  4. Spring依赖注入的三种方式

    看过几篇关于Spring依赖注入的文章,自己简单总结了一下,大概有三种方式: 1.自动装配 通过配置applicationContext.xml中的标签的default-autowire属性,或者标签 ...

  5. Spring自动注入的几种方式

    ---恢复内容开始---   @Service("accountEmailService")public class AccountEmailServiceImpl impleme ...

  6. Spring基础05——Spring依赖注入的三种方式

    Spring支持3种依赖注入的方式:属性注入.构造器注入.工厂 1.属性注入 属性注入即通过setter方法注入Bean的属性或依赖的对象.使用<property>元素,使用name属性指 ...

  7. spring——依赖注入的三种方式

    1 构造器注入(与构造器有直接关系) 默认无参构造 3种构造方式:通过<contructor-arg>调用类中的构造器 下标 <bean id="userService&q ...

  8. Spring——自动装配的三种实现方式

    依赖注入的本质是装配,装配是依赖注入的具体行为 spring会在上下文中自动寻找,并自动给bean装配属性 自动装配的三种方式 (1).在xml中显式的装配 (2).在java中显式的装配 (3).隐 ...

  9. spring生成EntityManagerFactory的三种方式

    spring生成EntityManagerFactory的三种方式 1.LocalEntityManagerFactoryBean只是简单环境中使用.它使用JPA PersistenceProvide ...

随机推荐

  1. unittest生成报告

    # html报告文件路径    report_abspath = os.path.join(report_path, "result.html")    fp = open(rep ...

  2. 对数据集做标准化处理的几种方法——基于R语言

    数据集——iris(R语言自带鸢尾花包) 一.scale函数 scale函数默认的是对制定数据做均值为0,标准差为1的标准化.它的两个参数center和scale: 1)center和scale默认为 ...

  3. 谈谈mybatis中的#与$的区别

    #相当于对数据 加上 双引号,$相当于直接显示数据 . #将传入的数据都当成一个字符串,会对自动传入的数据加一个双引号.如:", 如果传入的值是id,则解析成的sql为order by &q ...

  4. Web基础之http协议

    第6章 Web基础之http协议 第6章 Web基础之http协议一.http协议介绍 1.1)什么是超文本 1.2)什么是URL 1.3)什么是超文本传输协议二.访问网站分析三.页面请求信息解析(仅 ...

  5. Python自动化学习--鼠标和键盘事件

    from selenium import webdriver from selenium.webdriver import ActionChains import time driver = webd ...

  6. JavaScript设计模式 样例二 —— 策略模式

    策略模式(Strategy Pattern): 定义:定义了一族算法: 封装了每个算法: 这族的算法可互换代替. 目的:将算法的使用与算法的实现分离开来. 场景:可用来消除大量的条件分支语句. 例:J ...

  7. Linux安装php扩展memcache

    Linux安装php扩展memcache   php扩展memcache的作用是为了支持memcached数据库缓存服务器,下面是安装方法.1.下载并解压memcache文件 wget -c http ...

  8. 【Luogu4221】[WC2018] 州区划分

    题目链接 题目描述 略 Sol 一个州合法就是州内点形成的子图中 不存在欧拉回路(一个点也算欧拉回路). 这个东西显然就状压 dp 一下: 设 \(f[S]\) 表示当前考虑了 \(S\) 这个集合内 ...

  9. SQL Server死锁问题:事务(进程 ID x)与另一个进程被死锁在 锁 | 通信缓冲区资源上并且已被选作死锁牺牲品。请重新运行该事务。

    ### The error occurred while setting parameters### SQL: update ERP_SCjh_zzc_pl set IF_TONGBU=1 where ...

  10. 超赞的Linux软件分享(持续更新)

    开发 Android studio - Android 的官方 IDE:Android Studio 提供在各种类型的安卓设备上构建应用最快的工具. Aptana - Aptana Studio 利用 ...