面向对象设计(OOD)有助于我们开发出高性能、易扩展以及易复用的程序。当中。OOD有一个重要的思想那就是依赖倒置原则(DIP),并由此引申出IoC、DI以及Ioc容器等概念。

本文首先用实例阐述四个概念。而且给出Java版本号的演示样例代码。

依赖倒置原则(DIP)

依赖倒置是一种软件架构设计的原则,。依赖倒置原则,它转换了依赖,高层模块不依赖于低层模块的实现,而低层模块依赖于高层模块定义的接口。通俗的讲,就是高层模块定义接口,低层模块负责实现。

怎样理解呢?举例说明吧。

先看生活中的一个样例。

图1  ATM与银行卡

相信大部分取过钱的朋友都深有感触,仅仅要有一张卡,随便到哪一家银行的ATM都能取钱。在这个场景中。ATM相当于高层模块,而银行卡相当于低层模块。ATM定义了一个插口(接口),供全部的银行卡插入使用。也就是说,ATM不依赖于详细的哪种银行卡。

它仅仅需定义好银行卡的规格參数(接口)。全部实现了这样的规格參数的银行卡都能在ATM上使用。现实生活如此,软件开发更是如此。

依赖倒置(高层模块定义接口,低层模块负责实现)

在这个图中,我们发现高层模块定义了接口,将不再直接依赖于低层模块。低层模块负责实现高层模块定义的接口。

这样,当有新的低层模块实现时,不须要改动高层模块的代码。

由此。我们能够总结出使用DIP的长处:

系统更柔韧:能够改动一部分代码而不影响其它模块。

系统更健壮:能够改动一部分代码而不会让系统崩溃。

系统更高效:组件松耦合。且可复用,提高开发效率。

控制反转(IoC)

DIP是一种 软件设计原则,它只告诉你两个模块之间应该怎样依赖,可是它并没有告诉怎样做。IoC则是一种 软件设计模式,它告诉你应该怎样做,来解除相互依赖模块的耦合。控制反转(IoC),它为相互依赖的组件提供抽象,将依赖(低层模块)对象的获得交给第三方(系统)来控制。即依赖对象不在被依赖模块的类中直接通过new来获取。

依赖注入(DI)

控制反转(IoC)一种重要的方式。就是将依赖对象的创建和绑定转移到被依赖对象类的外部来实现。主要实现有三种方式:构造函数注入;属性注入;接口注入。

以下以将订单存储到数据库为例,用Java演示样例来说明问题。

如果系统开发初期使用SqlServer。我们简历数据库操作类。

public class SqlServer {

public void add(){

System.out.println("Addorder to Sql Server!");

}

}

定义一个order类:

public class Order {

private SqlServer ss=newSqlServer();

public void add(){

ss.add();

}

}

測试一下:

public class Main {

public static void main(String[]args) {

Order o=new Order();

o.add();

}

}

输出:

到这里没问题,可是假设须要加入MySql的数据库呢?

这里须要建立操作MySql的数据库类:

public class MySql {

public void add(){

System.out.println("Add to mysqlServer");

}

}

订单类须要改动:

public class Order {

private MySql ss=new MySql();

public void add(){

ss.add();

}

}

測试输出:

假设还须要很多其它的数据库操作呢?这样下去还需改动代码。组件之间高度耦合,可扩展性较差,它违背了DIP原则。

依赖注入就是解决上述问题。

依赖注入主要有三种实现方式:构造函数注入;属性注入;接口注入。以下用演示样例进行说明。

1、        构造函数

构造函数函数注入。毫无疑问通过构造函数传递依赖。因此,构造函数的參数必定用来接收一个依赖对象。那么參数的类型是什么呢?详细依赖对象的类型?还是一个抽象类型?依据DIP原则,我们知道高层模块不应该依赖于低层模块。两者应该依赖于抽象。那么构造函数的參数应该是一个抽象类型。

定义接口:

public interface IDB {

publicvoid add();

}

然后在SqlServer中实现这个接口

publicclass SqlServer implements IDB{

@Override

public void add() {

// TODO Auto-generatedmethod stub

System.out.println("Addinto Sql Server");

}

}

改动order类,在构造函数中注入对象:

public class Order {

private IDB idb;

public Order(IDB idb){

this.idb=idb;

}

public void add(){

this.idb.add();

}

}

測试类:

public class Main {

publicstatic void main(String[] args) {

IDBidb=new SqlServer();

Ordero=new Order(idb);

o.add();

}

}

输出:

通过构造函数注入的方式我们将依赖对象SqlServer的创建和绑定转移到了Order类的外部来实现。这样就解除了SqlServe和Order类的讴歌关系。当我们须要换成MySql数据库的时候。仅仅须要又一次定义一个Mysql类实现IDB接口,在用到的地方新建。在Order的外部又一次绑定依赖。不须要改动Order的代码。

比如:

新建MySql类:

public class MySql implements IDB{

@Override

publicvoid add() {

//TODO Auto-generated method stub

System.out.println("Addinto MySql server!");

}

}

在Main函数中仅仅须要需改:

IDB idb=new MySql();

測试:

2、        属性注入

属性注入是通过属性来传递依赖。

因此,我们首先须要在依赖类Order中定义一个属性。

public class Order{

private IDB idb;//私有属性

public void add(){

this.idb.add();

}

public IDB getIdb() {

return idb;

}

public void setIdb(IDB idb) {

this.idb = idb;//接受依赖

}

}

接口类以及数据库訪问类同上。Main方法须要这么写:

public class Main{

public static void main(String[] args) {

IDB idb=new SqlServer();

Order o=new Order();

o.setIdb(idb);

o.add();

}

}

測试输出:

Add into SqlServer

假设须要扩展MySql数据库。仅仅须要在Main方法里改动为:

IDBidb=newMySql();

測试输出:

Add into MySql

3、        接口注入

相比构造函数注入和属性注入。接口注入显得有些复杂,使用也不常见。详细思路是先定义一个接口。包括一个设置依赖的方法。然后依赖类。继承并实现这个接口。

首先定义一个接口:

public interface IDependent {

publicvoid setDependency(IDB idb);

}

依赖类实现这个接口:

public class Order implements IDependent{

privateIDB idb;

@Override

publicvoid setDependency(IDB idb) {

//TODO Auto-generated method stub

this.idb=idb;

}

publicvoid add(){

this.idb.add();

}

}

还须要定义数据库訪问的接口:

public interface IDB {

publicvoid add();

}

详细数据库的实现须要实现这个接口:

public class SqlServer implements IDB{

@Override

publicvoid add() {

//TODO Auto-generated method stub

System.out.println("Addinto Sql Server");

}

}

測试类:

public class Main {

publicstatic void main(String[] args) {

Ordero=new Order();

IDBidb=new SqlServer();

o.setDependency(idb);

o.add();

}

}

測试输出:

Add into SqlServer

假设须要加入MySql的訪问须要定义MySql的实现:

public class MySql implements IDB{

@Override

publicvoid add() {

//TODO Auto-generated method stub

System.out.println("Addinto MySql");

}

}

在測试类仅仅须要作例如以下改动就可以:

IDBidb=newMySql();

測试输出:

Add into MySql。

IoC容器

简单的来讲就是抛弃手动的方式创建依赖对象传递给被依赖模块而选取DI框架替我们创建来减轻我们的工作量。对于大型项目来说,相互依赖的组件比較多。假设还用手动的方式。自己来创建和注入依赖的话,显然效率非常低,并且往往还会出现不可控的场面。正因如此。IoC容器诞生了。IoC容器实际上是一个DI框架,它能简化我们的工作量。它包括下面几个功能:

l  动态创建、注入依赖对象。

l  管理对象生命周期。

l  映射依赖关系。

Java中比較突出的就是Spring了。很多其它Spring的介绍将出如今后面的博文里。

不可不知的DIP、IoC、DI以及IoC容器的更多相关文章

  1. Atitit。如何实现dip, di ,ioc ,Service Locator的区别于联系

    Atitit.如何实现dip, di ,ioc  ,Service Locator的区别于联系 1. Dip原则又来自于松耦合思想方向1 2. 要实现dip原则,有以下俩个模式1 3. Ioc和di的 ...

  2. dip vs di vs ioc

    https://stackoverflow.com/questions/6766056/dip-vs-di-vs-ioc https://docs.microsoft.com/en-us/aspnet ...

  3. Spring4学习回顾之路02—IOC&DI

    IOC&DI介绍 ●IOC:(Inversion of Control) :控制反转(反向获取资源) 其思想是反转资源获取的方向.传统的资源上查找方式要求组件向容器发起请求查找资源,作为回应, ...

  4. 深入理解DIP、IoC、DI以及IoC容器

    摘要 面向对象设计(OOD)有助于我们开发出高性能.易扩展以及易复用的程序.其中,OOD有一个重要的思想那就是依赖倒置原则(DIP),并由此引申出IoC.DI以及Ioc容器等概念.通过本文我们将一起学 ...

  5. 对依赖倒置原则(DIP)及Ioc、DI、Ioc容器的一些理解

    1.概述 所谓依赖倒置原则(Dependence Inversion Principle)就是要依赖于抽象,不要依赖于具体.简单的说就是要求对抽象进行编程,不要对实现进行编程,这样就降低了客户与实现模 ...

  6. 深入理解DIP、IoC、DI以及IoC容器(转)

    深入理解DIP.IoC.DI以及IoC容器 摘要 面向对象设计(OOD)有助于我们开发出高性能.易扩展以及易复用的程序.其中,OOD有一个重要的思想那就是依赖倒置原则(DIP),并由此引申出IoC.D ...

  7. 【转】深入理解DIP、IoC、DI以及IoC容器

    原文链接:http://www.cnblogs.com/liuhaorain/p/3747470.html 前言 对于大部分小菜来说,当听到大牛们高谈DIP.IoC.DI以及IoC容器等名词时,有没有 ...

  8. 对依赖倒置原则(DIP)及Ioc、DI、Ioc容器的一些理解(转)

    所谓依赖倒置原则(Dependence Inversion Principle)就是要依赖于抽象,不要依赖于具体.简单的说就是要求对抽象进行编程,不要对实现进行编程,这样就降低了客户与实现模块间的耦合 ...

  9. DIP、IoC、DI以及IoC容器

    深入理解DIP.IoC.DI以及IoC容器 摘要 面向对象设计(OOD)有助于我们开发出高性能.易扩展以及易复用的程序.其中,OOD有一个重要的思想那就是依赖倒置原则(DIP),并由此引申出IoC.D ...

随机推荐

  1. Eclipse用法和技巧九:自动添加try/catch块2

    上一篇介绍了如何给未检查异常快速增加try/catch语句,这里在补充一点其他相关操作.有时候我们增加了try/catch之后还需要在加一个finally块,比如android上每次分配一个curso ...

  2. OCP-1Z0-042-V12.39-47题

    47.Which two database operations can be performed at the mount stage of database startup? 题目解析: A和E在 ...

  3. HDU3709:Balanced Number(数位DP+记忆化DFS)

    Problem Description A balanced number is a non-negative integer that can be balanced if a pivot is p ...

  4. span设置固定宽度

    <span> 标签是被用来组合文档中的行内元素.相信对一般的网页设计师来讲是非常熟悉的朋友了,使用相当频繁,但我们往往很少对SPAN设定样式,一般也没什么必要,大多数都留给DIV老朋友了. ...

  5. Selenium WebDriver TestNg Maven Eclipse java 简单实例

    环境准备 前提条件Eclipse 已经安装过 TestNg ,Maven 插件 新建一个普通的java项目 点击右键 configure->convert to Maven Project 之后 ...

  6. 慎得慌二u赫然共和任务i个屁

    http://www.huihui.cn/share/8424421 http://www.huihui.cn/share/8424375 http://www.huihui.cn/share/842 ...

  7. SIP for android

    SIP for android   会话发起协议 Android提供了一个支持会话发起协议(SIP)的API,这可以让你添加基于SIP的网络电话功能到你的应用程序.Android包括一个完整的 SIP ...

  8. Flex Label自动截取、自动换行

    label.maxDisplayedLines=0;      // 默认多行显示,不截取 label.maxDisplayedLines=1;     //任意整数,显示单行文本,自动截取(...) ...

  9. 静态书架和js模拟翻书效果

    书籍图片随便找了个,有点难看,须要的自己替换个好看点的png格式图片 源代码下载:http://download.csdn.net/detail/sweetsuzyhyf/7604091

  10. 菜鸟版JAVA设计模式—从买房子看代理模式

    今天学习了代理模式. 相对于适配器模式,或者说装饰器模式,代理模式理解起来更加简单. 代理这个词应该比較好理解,取代去做就是代理. 比方,我们买卖房子,那么我们会找中介,我要卖房子,可是我们没有时间去 ...