Spring使用 --- 基本概念(一):DI,依赖注入

本文讲述sprint基本概念之一: DI, 即依赖注入。

什么是依赖注入

说类A依赖于类B,最简单的例子是类A有一个类型为类B的成员变量。

依赖注入是指类A不用关心类B对象如何创建,它只知道有一个类B的对象,只需要使用就行了。 这样有几个好处:

  1. 可以很容易替换类B的对象,达到不同的效果
  2. 类A更容易测试。只需要创建一个类B的mock对象即可。
  3. 类A与类B解藕

不使用依赖注入时,类A负责创建类B的对象,写法如下:

class A {
    B b;
    A (){
        // 此处A创建一个B的对象,作为自己的成员变量
        b = new B();
    }
    void foo(){
        b.bar();
    }
}

class B {
    void bar(){
        System.out.println("B bar");
    }
}

其中创建类B对象的代码写死在类A的定义中,很码很僵化。如果想使用B的一个子类对象,除了修改类A的代码,别无它法。

一种更好的方法是通过构造器传入类B的对象,代码如下:

class A {
    B b;
    A (B b){
        this.b = b;
    }
    void foo(){
        b.bar();
    }
}
// B的定义不变

这种方法也叫构造器注入。此时代码灵活了很多。此时类A的代码可以处理所有类B的子类。一般将B定义为接口,实际传入的是一个实现。这也是面向接口编程的一个例子。

Sprint中对于依赖注入实现的效果与构造器注入大致相同,只不过使用xml配置文件来完成将类B的对象装配给类A的对象。依赖注入主要维护对象之前的依赖关系,对象的创建。如果对象a依赖于另一个对象b,则Spring创建对象a时,根据事先定义的依赖规则(一般写在xml文件中),会先创建一个对象b,然后再创建对象a,最后将b装配给对象a。装配的意思是赋值,一般通过调用setter函数。

怎样使用

仍然采用上面的例子,类A,类B的定义如下:

class A {
    B b;
    A (B b){
        this.b = b;
    }
    void foo(){
        b.bar();
    }
}
class B {
    void bar(){
    }
}

再创建一个xml文件,指定类A、B的依赖关系:

<beans>
  <bean id="a" class="A">
    <constructor-arg ref="b"/>
  </bean>
  <bean id="b" class="B">
  </bean>
</beans>

最后在主代码中调用ApplicationContext.getBean("a")函数来获取类A的一个实例:

import org.springframework.context.support.ClassPathXmlApplicationContext;
class Test {
    public static void main(String[] args) {
        ClassPathXmlApplicationContext context =
            new ClassPathXmlApplicationContext("a.xml");
        A a = context.getBean("a");
        a.foo();
        context.close();
    }
}

其中ApplicationContext是bean的container,bean是一个符合一定要求的java类。

什么时候使用

只要是有对象依赖的情况下,都可以使用依赖注入来将类之前的依赖也类对象的创建这部分从核心代码里剥离出来,通过srping来管理及实现。感觉实际上是软件模块、组件更细的划分。

任何稍大一些的软件都是由多个类组成,这些类协同共同完成功能(通过组合的方式)。因此在绝大多数项目中都需要使用。

好处

  • 对象之间解藕
  • 可能方便的更改使用哪个依赖对象,只需要修改配置文件就行了
  • 类的测试会方便很多,只要为所有依赖的类创建mock对象即可

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏帮你学MatLab

《Experiment with MATLAB》读书笔记(一)

读书笔记(一) 这是第一部分——迭代 将代码复制到m文件中即可运行 % 迭代是计算的关键 % % 上键:调用先前的命令 % %下面这个“双%”表示一个se...

2898
来自专栏菩提树下的杨过

利用ganymed-ssh2远程执行其它Linux机器上的shell命令

实际应用中,有时候需要从web管理界面上,远程去启动其它linux主机上的程序,利用ssh协议可以方便的满足这一需求。事实上hadoop架构中,从nn上启动dn...

23110
来自专栏北京马哥教育

Linux Shell脚本面试25问

Q:1 Shell脚本是什么、它是必需的吗? 答:一个Shell脚本是一个文本文件,包含一个或多个命令。作为系统管理员,我们经常需要使用多个命令来完成一项任务...

3465
来自专栏H2Cloud

TCPDUMP 抓包

  写了个脚本, 用于调试服务器消息传输, 代码如下: #!/bin/bash if [ $# -eq 0 ] ; then echo "usage local...

2735
来自专栏ShaoYL

Protocol(协议)

3867
来自专栏Python攻城狮

Pyramid中的request和response1.Request2.Multidict3.Response

官方API:pyramid.request、pyramid.response 和 pyramid.httpexceptions。

751
来自专栏Java架构沉思录

聊聊Java动态代理(上)

前言 在之前的文章《聊聊设计模式之代理模式》中,笔者为大家介绍了代理模式,在这里简单回顾一下。代理模式的作用是提供一个代理来控制对一个对象的访问,因此我们可以...

35913
来自专栏编程坑太多

Python 读写文本(open)

1403
来自专栏数据结构与算法

7617:输出前k大的数

7617:输出前k大的数 查看 提交 统计 提问 总时间限制:10000ms单个测试点时间限制:1000ms内存限制:65536kB描述 给定一个数组,统计前k...

3745
来自专栏nummy

__import__详解

当使用import导入Python模块的时候,默认调用的是__import__()函数。直接使用该函数的情况很少见,一般用于动态加载模块。

822

扫码关注云+社区