Java程序员必须掌握的Spring依赖管理原理

Spring依赖注入

依赖注入(Dependency Injection)的意思就是对象通过构造器函数参数,工厂方法的参数,或者成员属性,定义了对象的依赖对象;容器在创建该对象时会负责注入这些依赖。这个过程是控制反转的,即不是由即将创建的对象来管理自己的依赖的发现和实例化,而是有Spring容器来实现。

在Spring中依赖注入有两种形式,第一种就是基于构造函数的注入,即通过调用构造函数,传入参数,也就是依赖来完成整个依赖注入流程;第二种就是基于setter方法的注入

构造函数的参数的匹配,要避免歧义,如指定类型,指定参数的次序等。如果是按照参数名字匹配,则必须开启debug模式进行编译,否则参数名字是不保留的。如果不想开启debug模式编译,则可以使用@ConstructorProperties注解
setter方法注入是先调用没有参数的默认构造函数构建对象,或者没有参数的静态工厂方法,实例化bean后,调用setter方法来将该对象注入。

通过使用依赖注入,可以使代码更简洁,更好地实现对象之间解耦。另外,通过依赖注入管理的的对象是POJO类,可以更好地进行测试。

如何选择合适的依赖注入方法?

最佳实践是通过构造器方法注入主要依赖对象,通过setter方法注入可选的依赖对象。虽然可以在setter方法上加上@Required注解来实现主要依赖对象注入,但一般还是推荐使用构造器注入必须的依赖。

使用构造器注入,可以使得应用的组件作为不可变的对象,而且可以保证注入依赖是非null的。另外,构造器注入返回的是一个完整的初始状态的实例。但是,一般不推荐大量使用构造方法注入,如果出现这种情况,则说明代码需要重构。

setter方法适合注入可选的依赖,这些依赖可能有默认值,而且在其他位置使用这些依赖时务必要进行null值检查。使用setter方法的一个好处是可以修改或者重新配置,或者需要时再注入。如基于JMX MBean的管理。

Spring依赖解析流程

首先ApplicationContext会被创建和初始化,会加载包括描述所有bean的元数据。这些配置元数据可以通过XML,Java代码或者注解来指定。

对于每一个bean,它的依赖表现形式是成员属性,构造器参数,或者静态工厂方法的参数。在bean真正创建时,Spring容器会提供这些依赖的对象。这些参数可能是需要设置的默认值,也可能是另外一个bean的引用。

Spring容器会验证每个bean的配置信息。并且在bean真正创建时才设置设置属性值或者参数值。
在Spring中,单例作用域的bean会提前初始化,在Spring容器创建时就进行了实例化。对于其他的作用域的bean,则只在需要时才进行创建。之所以单例作用域的bean会被提前初始化,主要是为了解决依赖检查的问题,下文的循环依赖一节会详细说明。

在Spring内部会构建一个创建bean的依赖图,按照这依赖关系来创建Bean。

循环依赖解决

如果使用构造函数注入,则不能有循环依赖的情况。如A构造器依赖B,同时B也构造器依赖A。Spring IoC容器会在运行时检测到循环依赖,抛BeanCurrentlyInCreationException异常。一种解决办法是通过setter方法来解决循环依赖的情况。

Spring会在容器加载时检测配置问题,如引用不存在或者循环依赖。Spring会在必要时才解析依赖,即尽可能晚的来解析依赖关系。延迟解析依赖可能导致后期请求获取对象时报错,如抛出一个异常,如丢失指定对象或者属性。这种配置的延迟的可见性导致的问题使得ApplicationContext的实现要求单例作用域的bean提前记性初始化。虽然会耗费内存和时间,因为并不是按需创建这些单例作用域的bean,但是可以在ApplicationContext创建时就可以发现配置问题。

下文会介绍通过指定bean的可以通过配置来覆盖默认的行为,使得单例作用域的bean也是延迟初始化。
如果没有循环依赖存在,则在注入依赖对象时,这些依赖的对象就已经初始化完成了。即如果A依赖B,则在A初始化时,B已经初始化完成了。也就是说,Bean是在相关依赖设置完成,并且相关的生命周期方法调用完毕后,才算是完成了初始化。

bean的延迟初始化

默认情况下ApplicationContext是提前初始化单例作用域的bean,作为ApplicationContext初始化的一部分。这样可以尽快的发现配置问题。可以通过指定bean的lazy-init="true",让bean在需要时才被初始化。

自动注入依赖

在Spring中可以自动注入依赖,可以减少指定属性或者构造器参数,还可以随着配置对象的变化来更新注入的对象。
自动注入依赖的模式有:通过名称注入,通过类型注入,和通过构造器注入

总结

本文总结了Spring中的依赖管理的基本原理和常见的问题,具体的依赖注入配置语法还需要参考Spring的官方文档来进行。

原文地址:https://blog.51cto.com/14230003/2381524

时间: 2024-07-29 14:24:45

Java程序员必须掌握的Spring依赖管理原理的相关文章

【软帝学院】Spring Boot 是什么?为什么说java程序员必须要掌握Spring Boot ?

Spring Boot 是什么?为什么说java程序员必须要掌握Spring Boot ? 我们从Spring Boot 诞生的背景是什么?Spring 企业又是基于什么样的考虑创建 Spring Boot? 企业使用 Spring Boot会带来什么样变化?来分析 Spring 历史 说起 Spring Boot 我们不得不先了解一下 Spring 这个企业,不仅因为 Spring Boot 来源于 Spirng 大家族,而且 Spring Boot 的诞生和 Sping 框架的发展息息相关.

一道78%的Java程序员搞不清的Spring bean面试题

熟悉Spring开发的朋友都知道Spring提供了5种scope分别是singleton.prototype.request.session.global session.如下图是官方文档上的截图,感兴趣的朋友可以进去看看这五种分别有什么不同.今天要介绍的是这五种中的前两种,也是Spring最初提供的bean scope singleton 和 prototype.Spring官方文档介绍如下图: 单例bean与原型bean的区别 如果一个bean被声明为单例的时候,在处理多次请求的时候在Spr

Java程序员须知的七个日志管理工具(转)

Splunk vs. Sumo Logic vs. LogStash vs. GrayLog vs. Loggly vs. PaperTrails vs. Splunk>Storm 英文原文:The 7 Log Management Tools You Need To Know 日志管理工具有 Splunk.Sumo Logic.LogStash.GrayLog.Loggly 和 PaperTrails 等等,数不胜数.日志就像石油,二十多年了我们一直想摆脱它,却一直没有做到. 为了处理日益增长

推荐给 Java 程序员的 7 本书

< Java 编程思想> 适合各个阶段 Java 程序员的必备读物.书中对 Java 进行了详尽的介绍,与其它语言做了对比,解释了 Java 很多特性出现的原因和解决的问题.初学者可以通过此书快速掌握 Java 面向对象的理念,学会正确使用 Java 的各种特性:平时开发中可以将此书作为工具书参考,遇到疑难问题或查缺补漏都可以参考此书:有经验的开发者重温此书,可以加深对 Java 的理解,开发能力再上一层楼. <设计模式> 四位作者均是国际公认的面向对象软件领域的专家.此书以 C+

java程序员培训:为什么要用Spring含三大框架简介

文章来源:http://www.zretc.com/technologyDetail/476.html Spring作为三大框架之一,一直困惑她有什么用?为什么要使用Spring呢?(大部分来自网上,引用出处太多,恕不一一标明.). SSH在J2EE项目中表示了3种框架,即 Spring + Struts +Hibernate. Struts对Model,View和Controller都提供了对应的组件.Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架,它由Rod Jo

Java程序员应该了解的10个面向对象设计原则

面向对象设计原则: 是OOPS(Object-Oriented Programming System,面向对象的程序设计系统)编程的核心,但大多数Java程序员追逐像Singleton.Decorator.Observer这样的设计模式,而不重视面向对象的分析和设计.甚至还有经验丰富的Java程序员没有听说过OOPS和SOLID设计原则,他们根本不知道设计原则的好处,也不知道如何依照这些原则来进行编程. 众所周知,Java编程最基本的原则就是要追求高内聚和低耦合的解决方案和代码模块设计.查看Ap

Java程序员面试题集(71-85)

Java程序员面试题集(71-85) 摘要:这一部分主要包括了UML(统一建模语言).面向对象的设计原则(六原则一法则).GoF设计模式.企业级设计模式.JDBC(Java数据库连接).XML(可扩展标记语言)等知识. 71.UML是什么?UML中有哪些图? 答:UML是统一建模语言(Unified Modeling Language)的缩写,它发表于1997年,综合了当时已经存在的面向对象的建模语言.方法和过程,是一个支持模型化和软件系统开发的图形化语言,为软件开发的所有阶段提供模型化和可视化

(转载)Java程序员应当知道的10个面向对象设计原则

面向对象设计原则是OOPS编程的核心, 但我见过的大多数Java程序员热心于像Singleton (单例) . Decorator(装饰器).Observer(观察者) 等设计模式,而没有把足够多的注意力放在学习面向对象的分析和设计上面.学习面向对象编程像"抽象"."封装"."多态"."继承" 等基础知识是重要的,但同时为了创建简洁.模块化的设计,了解这些设计原则也同等重要.我经常看到不同经验水平的java程序员,他们有的不知

【转】你离顶尖 Java 程序员,只差这11本书的距离

个人认为看书有两点好处: 能出版出来的书一定是经过反复的思考.雕琢和审核的,因此从专业性的角度来说,一本好书的价值远超其他资料 对着书上的代码自己敲的时候方便 "看完书之后再次提升自我的最好途径是看一些相关的好博文",我个人认为这是学习的第二步,因为一本书往往有好几百页,好的博文是自己看书学习之后的一些总结和提炼,对于梳理学习的内容很有好处,当然这里不是说自己的学习方法,就不再扯下去了. 很多程序员们往往有看书的冲动,但不知道看哪些书,下面我就给各位Java程序猿们推荐一些好书(每本书