二、依赖注入的应用模式

前面我们了解了依赖注入的基本概念,也对一些依赖注入框架进行了简单的介绍,这一章我们主要来讨论作为开发者如何利用依赖注入框架来实现依赖注入的设计思想。

1.     依赖注入的方式

前面我们提到,所谓“依赖”,最简单地去解释就是一个Java类里的成员变量。我们都知道,给一个类中的私有成员变量赋值的方法通常有:通过Constructor构造方法、通过Setter方法、通过反射机制将私有变量的可见性设为true这三种方法。同样道理,依赖注入框架也是利用这三种方式来完成依赖对象的设定的,我们分别称之为Constructor注入、Setter注入、成员变量注入。

这里有两点需要注意的地方。一是除了这三种常见的注入方式,还有一些其他的注入方式,例如Spring框架中提供的lookup-method为代表的方法注入模式,也被称作“装饰器”模式,还有像Guice中的方法参数注入等等,我们在此不再详加讨论。二是并不是所有的依赖注入框架都提供这三种常见的注入方式,比如Seam2.x版本里就不支持Constructor注入的方式。

1.1.    Constructor注入

利用Constructor注入,即在依赖者类的构造函数上声明注入点,而要被注入的对象则是构造函数的参数。

我们来看利用Spring框架的例子。首先我们将Bank和DepositBook两个依赖类声明为Spring容器管理的组件:

@Component // 声明此依赖为Spring组件

public class BankICBC implements Bank { // ……  }

@Component // 声明此依赖为Spring组件

public class DepositBookICBC implements DepositBook { // ……  }

这里我们采用Spring 3.x提供的@Component方式来声明Spring组件,当然还有很多其它的方式,就不再详细阐述了。将这两个依赖声明为Spring容器管理的组件之后,它们就可以在被需要的时候由容器提供给依赖者了。例如在储户Depositor类中,我们定义了以Constructor为注入点的依赖请求:

@Component

public class Depositor {

@Autowired  // Constructor注入点

public Depositor(Bank bank, DepositBook depositBook) {

this.bank = bank;

this.depositBook = depositBook;

}

private Bank bank;

private DepositBook depositBook;

public Cash withDraw(BigDecimal amount) {

return bank.withDraw(depositBook, amount);

}

}

之后主程序在调用这个Depositor类的时候,容器就会自动将保管的依赖对象设定好之后分发给主程序了。

public void runSpringConstructorDI() {

AnnotationConfigApplicationContext context =

new AnnotationConfigApplicationContext("tutorial.di.ch01");

Depositor depositor = context.getBean(Depositor.class);

depositor.withDraw(new BigDecimal(10000));

}

1.2.    Setter注入

第二种常用的声明注入点的方法是Setter注入,即依赖注入框架利用反射机制调用依赖者类的Setter方法来完成依赖注入。

我们来看利用Seam框架的例子。首先还是要将Bank和DepositBook两个依赖类声明为Seam容器管理的组件:

@Name("bank")

public class BankICBC implements Bank { // ……  }

@Name("depositBook")

public class DepositBookICBC implements DepositBook { // ……  }

之后在依赖者Depositor类中的Setter方法处声明注入点:

@Name("depositor")

public class Depositor {

private Bank bank;

private DepositBook depositBook;

@In  // bank的Setter注入点

public void setBank(Bank bank) {

this.bank = bank;

}

@In  // depositBook的Setter注入点

public void setDepositBook(DepositBook depositBook) {

this.depositBook = depositBook;

}

public Cash withDraw(BigDecimal amount) {

return bank.withDraw(depositBook, amount);

}

}

注意Seam框架缺省是将与该依赖字段同名的标识符所绑定的依赖对象注入进来,如果想注入与字段名不同的标识符得依赖对象,需要使用@In(“依赖对象标识符”)这样的语法。

1.3.    成员变量注入

成员变量注入是指依赖注入框架利用反射机制,直接将依赖者类的成员变量set为容器管理的依赖对象。

仍以Seam框架为例,与刚才Setter注入模式唯一不同的就是将@In标注在成员变量旁,而不是其Setter方法上。

@Name("depositor")

public class Depositor {

@In  // bank的成员变量注入点

private Bank bank;

@In  // depositBook的成员变量注入点

private DepositBook depositBook;

public Cash withDraw(BigDecimal amount) {

return bank.withDraw(depositBook, amount);

}

}

1.4.    注入模式的选择

对于该怎么使用前面介绍的这几种注入模式,究竟在什么情况下该使用哪种,这是一个讨论非常广泛的话题。这里只举一个简单的例子来说明,比如要设计一个不可变(final)依赖的类,则必须使用Constructor注入方式:

@Component

public class Depositor {

@Autowired  // Constructor注入点

public Depositor(Bank bank, DepositBook depositBook) {

this.bank = bank;

this.depositBook = depositBook;

}

private final Bank bank; // 不可变的依赖

private final DepositBook depositBook; // 不可变的依赖

public Cash withDraw(BigDecimal amount) {

return bank.withDraw(depositBook, amount);

}

}

再如如果一个类所拥有的依赖数量过多、而开发者又不想构造方法的参数成为“过长参数列表”(这是一种影响代码的可读性和可维护性的反模式,Martin Fowler于1999年在其《Refactoring:Improving the Design of Existing Code》一书中提出),则可考虑Setter注入或者成员变量注入。此外,如果遇到了类似“循环依赖”的情况,例如宿主和寄生动物这两个对象就属于相互依存的“循环依赖”模式,这样的情况下,我们就可能需要Constructor和Setter两种混合的模式才能解决。总而言之,选择依赖注入模式是一个很大的话题,我们在此就不再深入讨论了。

转载于:https://www.cnblogs.com/Free-Thinker/p/4173273.html

依赖注入及AOP简述(五)——依赖注入的方式 .相关推荐

  1. 依赖注入及AOP简述(一)——“依赖”的概念 .

    一.入门:依赖注入 作为一种全新的设计模式理念,"依赖注入"这个词汇在软件设计开发中已经是越来越耳熟能详了,而各种流行于开源社区的"依赖注入框架",也越来越多的 ...

  2. 依赖注入及AOP简述(六)——字符串请求模式 .

    2.     依赖注入对象的请求模式 前一节我们讨论了关于声明注入点的几种方法,这一节主要来介绍在注入点上如何定位到所需要的标识符的话题.基本上,我们可以用字符串为标识符来请求依赖对象.或者用全类名( ...

  3. 【C#|.NET】从控制反转(依赖注入)想到事件注入 (非AOP)

    前文 事件注入的想法是由依赖注入所联想到 依赖注入不算什么吸引人的话题 本篇就不详说了 不过有闲暇时间的机会不妨按照自己的兴趣去摸索.研究一些东西,也是一种乐子. 在抓虫系列里简单的描述一下依赖注入在 ...

  4. JavaEE开发之Spring中的依赖注入与AOP编程

    上篇博客我们系统的聊了<JavaEE开发之基于Eclipse的环境搭建以及Maven Web App的创建>,并在之前的博客中我们聊了依赖注入的相关东西,并且使用Objective-C的R ...

  5. ASP.NET Core中的依赖注入(2):依赖注入(DI)

    IoC主要体现了这样一种设计思想:通过将一组通用流程的控制从应用转移到框架之中以实现对流程的复用,同时采用"好莱坞原则"是应用程序以被动的方式实现对流程的定制.我们可以采用若干设计 ...

  6. everythingtoolbar.dll”或它的一个依赖项。_ASP.NET Core依赖注入最佳实践、提示和技巧...

    译者前言 本文译自ABP框架的开发博客<ASP.NET Core Dependency Injection Best Practices, Tips & Tricks>一文(原作者 ...

  7. 架构设计之依赖倒置、控制反转与依赖注入

    名词解释 依赖:一种模型元素之间的关系的描述.例如类A调用了类B,那么我们说类A依赖于类B. 耦合:一种模型元素之间的关系的描述.例如类A调用了类B或类B调用了类A,那么我们说类A与类B有耦合关系. ...

  8. spring注入私有字段_Spring字段依赖注入示例

    spring注入私有字段 了解如何编写Spring Field Injection示例 . 字段注入是Spring框架 依赖注入的一种 . 在本教程中,我们将编写几个类,并看一看现场注入工程. 有关S ...

  9. java 有参数的构造函数如何注入_Spring5参考指南:依赖注入

    依赖注入 依赖注入就是在Spring创建Bean的时候,去实例化该Bean构造函数所需的参数,或者通过Setter方法去设置该Bean的属性. Spring的依赖注入有两种基于构造函数的依赖注入和基于 ...

最新文章

  1. 四年上册级计算机教学计划,小学数学四年级上册教学计划
  2. java 编辑我的世界_Editing Java版1.13 (section)
  3. 运算符重载(c++细节篇二)
  4. Python3序列解包
  5. linux查看git是否运行,关于linux:使用脚本检查git分支是否领先于另一个
  6. 怎么安装Win10,硬盘安装Win10系统图文教程
  7. 属性值动态调整_这可能是你见过最牛的CAD粗糙度动态块了!
  8. Java基础-HashMap集合
  9. 【最短路径问题】计蒜客:圣诞树
  10. 怎么把3d风格的图变成2d_单图像三维重建、2D到3D风格迁移和3D DeepDream
  11. RK3288_Android7.1调试RTC总结(二)
  12. 使用Shader进行UGUI的优化
  13. λ^n |λI−AB|= λ^m |λI−BA|
  14. 福利大放送:空间统计插值大数据PPT
  15. python在经济学的应用_『经济学在读研究生学习python可以用来做什么』python经济应用教程...
  16. vps没有mysql怎么用商店_买了vps能干什么,何不用它搭建一个5T私人网盘?
  17. 一键清空朋友圈软件_一键清空朋友圈软件有哪些(推荐几个好用软件)
  18. Make Cents? Gym - 101350M (水)
  19. 微服务与敏捷开发(Scrum/Kanban)的核心思想之我见
  20. java语言就业方向_Java的就业方向有哪些?

热门文章

  1. Python Django单表增删改操作
  2. 请讲一下浏览器从接收到一个URL,到最后展示出页面,经历了哪些过程
  3. MySQL添加中文记录报错解决方法
  4. Spring 体系结构详解
  5. mysql php 变量赋值,mysql变量赋值要注意的_MySQL
  6. windows清理图标缓存并重新加载
  7. 动态折线图-Android篇
  8. 解决:pycharm中动画函数animation.FuncAnimation不起作用
  9. emui10有机会升级鸿蒙,鸿蒙OS+EMUI10,能否助力华为实现科技突围?
  10. Linux命令详解:./configure、make、make install 命令