上篇文章我们已经学习了1.4小结中关于依赖注入跟方法注入的内容。这篇文章我们继续学习这结中的其他内容,顺便解决下我们上篇文章留下来的一个问题-----注入模型

前言:

在看下面的内容之前,我们先要对自动注入及精确注入有一个大概的了解,所谓精确注入就是指,我们通过构造函数或者setter方法指定了我们对象之间的依赖,也就是我们上篇文章中讲到的依赖注入,然后Spring根据我们指定的依赖关系,精确的给我们完成了注入。那么自动注入是什么?我们看下面一段代码:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/dbeans/spring-beans.xsd"
><bean id="auto" class="com.dmz.official.service.AutoService" autowire="byType"/><bean id="dmzService" class="com.dmz.official.service.DmzService"/>
</beans>
public class AutoService {DmzService service;public void setService(DmzService dmzService){System.out.println("注入dmzService"+dmzService);service = dmzService;}
}
public class DmzService {}
public class Main03 {public static void main(String[] args) {ClassPathXmlApplicationContext cc =new ClassPathXmlApplicationContext("application.xml");System.out.println(cc.getBean("auto"));}
}

在上面的例子中我们可以看到

  1. 我们没有采用注解@Autowired进行注入

  2. XML中没有指定属性标签<property>

  3. 没有使用构造函数

但是,打印结果如下:


注入dmzServicecom.dmz.official.service.DmzService@73a8dfcc  // 这里完成了注入
com.dmz.official.service.AutoService@1963006a

可能细心的同学已经发现了,在AutoService的标签中我们新增了一个属性autowire="byType",那么这个属性是什么意思呢?为什么加这个属性就能帮我们完成注入呢?不要急,我们带着问题继续往下看。

自动注入:

这部分内容主要涉及官网中的1.4.5小结。

我们先看官网上怎么说的:

自动注入的优点:

大概翻译如下:

Spring可以自动注入互相协作的bean之间的依赖。自动注入有以下两个好处:

  • 自动注入能显著的减少我们指定属性或构造参数的必要。这个不难理解,我们在上篇文章中讲过了,依赖注入的两种方式,setter方法跟构造函数,见上篇文章依赖注入。在前言中的例子我们也能发现,我们并不需要指定属性或构造参数

  • 自动装配可以随着对象的演化更新配置。例如,如果需要向类添加依赖项,则可以自动满足该依赖项,而不需要修改配置。因此,自动装配在开发过程中特别有用,但是当我们的代码库变的稳定时,自动装配也不会影响我们将装配方式切换到精确注入(这个词是我根据官网阅读加自己理解翻译过来的,也就是官网中的(explicit wiring

注入模型:

接下来,官网给我们介绍了自动注入的四种模型,如图:

我们一一进行解析并测试:

  • no

这是目前Spring默认的注入模型,也可以说默认情况下Spring是关闭自动注入,必须要我们通过setter方法或者构造函数完成依赖注入,并且Spring也不推荐修改默认配置。我们使用IDEA时也可以看到

用红线框出来的部分建议我们使用精确的方式注入依赖。

从上面来说,Spring自动注入这种方式在我们实际开发中基本上用不到,但是为了更好的理解跟学习Spring源码,我们也是需要好好学习这部分知识的。

  • byName

这种方式,我们为了让Spring完成自动注入需要提供两个条件

  1. 提供setter方法

  2. 如果需要注入的属性为xxx,那么setter方法命名必须是setXxx,也就是说,命名必须规范

在找不到对应名称的bean的情况下,Spring也不会报错,只是不会给我们完成注入。

测试代码:

//记得需要将配置信息修改为:<bean id="auto" class="com.dmz.official.service.AutoService"      autowire="byName"/>public class AutoService {DmzService dmzService;/***  setXXX,Spring会根据XXX到容器中找对应名称的bean,找到了就完成注入*/public void setDmzService(DmzService dmzService){System.out.println("注入dmzService"+dmzService);service = dmzService;}
}

另外我在测试的时候发现,这种情况下,如果我们提供的参数不规范也不会完成注入的,如下:

public class AutoService {DmzService dmzService;// indexService也被Spring所管理IndexService indexService;/*** setXXX,Spring会根据XXX到容器中找对应名称的bean,找到了就完成注入*/public void setDmzService(DmzService dmzService,IndexService indexService) {System.out.println("注入dmzService" + dmzService);this.dmzService = dmzService;}
}

本以为这种情况Spring会注入dmzService,indexService为null,实际测试过程中发现这个set方法根本不会被调用,说明Spring在选择方法时,还对参数进行了校验,byName这种注入模型下,参数只能是我们待注入的类型且只能有一个

  • byType

测试代码跟之前唯一不同的就是修改配置autowire="byType",这里我们测试以下三种异常情况

  1. 找不到合适类型的bean,发现不报异常,同时不进行注入

  2. 找到了多个合适类型的bean,Spring会直接报错Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'com.dmz.official.service.DmzService' available: expected single matching bean but found 2: dmzService,dmzService2

  3. set方法中有两个参数,切两个参数都能找到唯一一个类型符合的bean,不报异常,也不进行注入

另外需要说明的是,我在测试的过程,将set方法仅仅命名为set,像这样public void set(DmzService dmzService),这种情况下Spring也不会进行注入

我们可以发现,对于这两种注入模型都是依赖setter方法完成注入的,并且对setter方法命名有一定要求(只要我们平常遵从代码书写规范,一般也不会踩到这些坑)。第一,不能有多个参数;第二,不能仅仅命名为set

  • constructor

当我们使用这种注入模型时,Spring会根据构造函数查找有没有对应参数名称的bean,有的话完成注入(跟前文的byName差不多),如果根据名称没找到,那么它会再根据类型进行查找,如果根据类型还是没找到,就会报错。

自动注入的缺陷:

这里不得不说一句,Spring官网在这一章节有三分之二的内容是在说自定注入的缺陷以及如何将一个类从自动注入中排除,结合默认情况下自动注入是关闭的(默认注入模型为no),可以说明,在实际使用情况中,Spring是非常不推荐我们开启自动注入这种模型的。从官网中我们总结自动注入有以下几个缺陷:

  • 精确注入会覆盖自动注入。并且我们不能注入基本数据类型,字符串,Class类型(这些数据的数组也不行)。而且这是Spring故意这样设计的

  • 自动注入不如精确注入准确。而且我们在使用自动注入时,对象之间的依赖关系不明确

  • 对于一些为Spring容器生成文档的工具,无法获取依赖关系

  • 容器中的多个bean定义可能会与自动注入的setter方法或构造函数参数指定的类型匹配。对于数组、集合或映射实例,这可能不会产生什么问题。但是,对于期望单个值的依赖项,我们无法随意确定到底有谁进行注入。如果没有唯一的bean定义可用,则会抛出异常

如何将Bean从自动注入中排除?

这里主要用到autowire-candidate这个属性,我们要将其设置为false,这里需要注意以下几点:

  1. 这个设置只对类型注入生效。这也很好理解,例如我们告诉Spring要自动注入一个indexService,同时我们又在indexService的配置中将其从自动注入中排除,这就是自相矛盾的。所以在byName的注入模型下,Spring直接忽略了autowire-candidate这个属性

  2. autowire-candidate=false这个属性代表的是,这个bean不作为候选bean注入到别的bean中,而不是说这个bean不能接受别的bean的注入。例如在我们上面的例子中我们对AutoService进行了如下配置:

 <bean id="auto" class="com.dmz.official.service.AutoService" autowire="byType" autowire-candidate="false"/>

代表的是这个bean不会被注入到别的bean中,但是dmzService任何会被注入到AutoService

另外需要说明的是,对于自动注入,一般我们直接在顶级的标签中进行全局设置,如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"<!--在这里进行配置-->default-autowire="byName">

自动注入跟精确注入的比较总结:

连同上篇文章依赖注入,我画了下面一个图:

  • 从关注的点上来看,自动注入是针对的整个对象,或者一整批对象。比如我们如果将autoService这个bean的注入模型设置为byName,Spring会为我们去寻找所有符合要求的名字(通过set方法)bean并注入到autoService中。而精确注入这种方式,是我们针对对象中的某个属性,比如我们在autoService中的dmzService这个属性字段上添加了@AutoWired注解,代表我们要精确的注入dmzService这个属性。而方法注入主要是基于方法对对象进行注入

  • 我们通常所说byName,byType跟我们在前文提到的注入模型中的byName,byType是完全不一样的。通常我们说的byName,byType是Spring寻找bean的手段。比如,当我们注入模型为constructor时,Spring会先通过名称找对符合要求的bean,这种通过名称寻找对应的bean的方式我们可以称为byName。我们可以将一次注入分为两个阶段,首先是寻找符合要求的bean,其次再是将符合要求的bean注入。也可以画图如下:

补充(1.4小结的剩余部分)

这部分比较简单,也是1.4小节中剩余的两个小知识,在这篇文章我们也一并学习了~

depends-on:

我们首先要知道,默认情况下,Spring在实例化容器中的对象时是按名称进行自然排序进行实例化的。比如我们现在有A,B,C三个对象,那么Spring在实例化时会按照A,B,C这样的顺序进行实例化。但是在某些情况下我们可能需要让B在A之前完成实例化,这个时候我们就需要使用depends-on这个属性了。我们可以通过形如下面的配置完成:

<bean id="a" class="xx.xx.A" depends-on="b"/>
<bean id="b" class="xx.xx.B" />

或者:

@Component
@DependsOn("b")
public class A {
}

lazy:

默认情况下,Spring会在容器启动阶段完成所有bean的实例化,以及一系列的生命周期回调。某些情况下,我们

可能需要让某一个bean延迟实例化。这种情况下,我们需要用到lazy属性,有以下两种方式:

  1. XML中bean标签的lazy-init属性

<bean id="lazy" class="com.something.ExpensiveToCreateBean" lazy-init="true"/>
  1. @Lazy注解

@Component
// 懒加载
@Lazy
public class A {}

到此为止,官网中1.4小节中的内容我们就全学习完啦!最核心的部分应该就是上文中的这个图了。我们主要总结了Spring让对象产生依赖的方式,同时对各个方式进行了对比。通过这部分的学习,我觉得大家应该对Spring的依赖相关知识会更加系统,这样我们之后学习源码时碰到疑惑也会少很多。

下面我们还要继续学习Spring的官网,比如前面文章提到的Beandefinition到底是什么东西?Spring中的Bean的生命周期回调又是什么?这些在官网中都能找到答案。

给自己加油,也给所有看到这篇文章的同学加油~!!

Spring官网阅读(三)自动注入相关推荐

  1. Spring官网阅读 | 总结篇

    接近用了4个多月的时间,完成了整个<Spring官网阅读>系列的文章,本文主要对本系列所有的文章做一个总结,同时也将所有的目录汇总成一篇文章方便各位读者来阅读. 下面这张图是我整个的写作大 ...

  2. Spring官网阅读(二)(依赖注入及方法注入)

    上篇文章我们学习了官网中的1.2,1.3两小节,主要是涉及了容器,以及Spring实例化对象的一些知识.这篇文章我们继续学习Spring官网,主要是针对1.4小节,主要涉及到Spring的依赖注入.虽 ...

  3. Spring官网阅读(一)容器及实例化

    从今天开始,我们一起过一遍Spring的官网,为Spring源码的学习打好基础.在这个过程中,不会涉及过多底层的代码,更多是通过例子证明我们在官网得出的结论,希望自己可以坚持下来,给自己加个油!!! ...

  4. Spring 官网阅读指南

    一.概述 Spring官网:https://spring.io/,界面如下: 二.各模块介绍 进入首页如上图,在首页官网会展示一些当前Spring比较流行的技术.可以看见导航栏有几个模块,信息如下: ...

  5. Spring官网阅读(四)BeanDefinition(上)

    前面几篇文章已经学习了官网中的1.2,1.3,1.4三小结,主要是容器,Bean的实例化及Bean之间的依赖关系等.这篇文章,我们继续官网的学习,主要是BeanDefinition的相关知识,这是Sp ...

  6. Spring官网阅读(九)Spring中Bean的生命周期(上)

    文章目录 生命周期回调 1.Bean初始化回调 2.Bean销毁回调 3.配置默认的初始化及销毁方法 4.执行顺序 5.容器启动或停止回调 Lifecycle 接口 LifecycleProcesso ...

  7. Spring官网和源码系列-如何阅读

    作为一名Java从业者,有听过这么一句话,Spring源码至少看三遍才能算是一名软件工程师.对于读源码,大家就觉得很头疼,潜意识里感觉枯燥.难,基本就是能不读就不读,工作中够用就行了.不读源码的确可以 ...

  8. Spring Boot入门三:创建Spring Boot项目;(包括【Spring 官网start.spring.io】方式,【IDEA集成的Spring Initializr】方式)

    说明: (1)介绍了两种创建Spring Boot项目的方式:[Spring 官网start.spring.io]方式,[IDEA集成的Spring Initializr]方式: (2)更推荐使用[I ...

  9. Spring官网学习(一)概述

    文章目录 1.Spring官网简介 2.Spring总览 2.1.什么是Spring 2.2.Spring的发展历程 3.Spring的设计理念 3.1.Spring的优点 4.IOC和AOP浅析 4 ...

最新文章

  1. 接口里面的方法都是抽象方法吗_正确的敷面膜方法 你真的都掌握了吗
  2. IOS开发笔记1-写一个hello world!程序
  3. 人家可是见过大世面的
  4. jeesite使用心得(二)
  5. 全新视角洞察租车行业发展趋势
  6. (14)Verilog数据类型-基本语法(二)(第3天)
  7. Caliburn.Micro学习笔记(四)----IHandleT实现多语言功能
  8. oracle怎么同步时间设置,【oracle数据库获取当前时间】
  9. 2006-8-11 11:29:00 搜索算法及其在ACM竞赛中的应用(作者/刘力科 计算机系01级4班)...
  10. python如何调用程序_Python中调用其他程序的方式详解
  11. 小白的web优化之路 一、使用redis来缓存信息
  12. kmeans python interation flag_Bisecting k-means聚类算法实现
  13. vmware中linux启动项,VMWare虚拟机中安装Linux系统并启用
  14. 数据库管理工具哪个好?强力推荐Navicat Premium 16 mac中文版
  15. 学人工智能有前途吗?人工智能前景-AI就业方向
  16. 微信小程序 ‘腾讯位置服务地图选点插件申请失败’
  17. P2600 [ZJOI2008]瞭望塔
  18. Cyclone V SoC FPGA学习之路第一章:综述
  19. 终端shell常用命令
  20. 数据分析-Tableau01-柱状图、堆积柱状图、直方图系列

热门文章

  1. matlab显示的图片,手动保存时四周有白边
  2. Python基础教程(十二):GUI编程、版本区别、IDE
  3. Gradle 设置 本地maven仓库及发布mavenLocal()路径的方法
  4. ASP.NET 4.0 的新特性(2 在WebForm中使用Routing)(翻译)
  5. 2017年阵亡创业公司圈钱魔咒 烧钱补贴买流量已经过时?
  6. Linux桌面极速文件浏览器PCMan
  7. caffe学习(三):caffe开发环境安装(Ubuntu)
  8. 拷贝构造函数的调用以及浅拷贝与深拷贝的理解
  9. pc显示器分辨率 前端_五款高性价比PC显示器推荐 499元起
  10. python数据库抓取并保存_python:微信消息抓取、转发和数据库存储及源码