Spring中的IoC容器和依赖注入简单入门

Spring实际上已经快成为了java后端开发标准的一部分。其框架结构如下图。


Spring Framework

理解Spring中IoC容器和依赖注入是理解Spring的基础。

1、类之间的依赖关系

类之间的依赖(dependency)是指一个类A使用到了另一个类B,所以,有些英文的材料中将这种关系表述为引用(using)。这种关系具有偶然性、临时性,也非常弱,但是,类B的变化会影响到类A。代码中,一般表现为,类A中的某个成员函数用到了某个类B类型的参数,或者类A中的某个成员函数,用到了类B类型的某个临时变量,等等。
在UML中,依赖关系用带箭头的虚线表示,箭头从使用类指向被依赖的类。如下:


依赖关系

如下的代码展示了一个类依赖关系的例子:

// Interface HelloWorld
public interface HelloWorld {
    public void sayHello();
}
 
// Class implements HelloWorld
public class SpringHelloWorld implements HelloWorld {
   public void sayHello()  {
           System.out.println("Spring say Hello!");
   }
}
 
// Other class implements HelloWorld
public class StrutsHelloWorld implements HelloWorld {
   public void sayHello()  {
           System.out.println("Struts say Hello!");
   }
}
 
 
// And Service class
public class HelloWorldService {
    
     // Field type HelloWorld
     private HelloWorld helloWorld;
    
     // Constructor HelloWorldService
     // It initializes the values for the field 'helloWorld'
     public HelloWorldService()  {
           this.helloWorld = new StrutsHelloWorld();
     }
 
}

或者
// And Service class
public class HelloWorldService {
    
     // Field type HelloWorld
     private HelloWorld helloWorld;
    
     // Constructor HelloWorldService
     // It initializes the values for the field 'helloWorld'
     public HelloWorldService()  {
           this.helloWorld = new SpringHelloWorld();
     }
 
}

上面例子的类图如下。


示例类图

生成该类图的PlantUML脚本如下:

@startuml

Title "类图"
interface HelloWorld
HelloWorld <|.. SpringHelloWorld:实现
HelloWorld <|.. StrutsHelloWorld:实现
HelloWorldService ..> HelloWorld:依赖

@enduml

2、反转控制和依赖注入

在上面的例子中,HelloWorldService类通过HelloWorld接口依赖于StrutsHelloWorld和SpringHelloWorld类。
当HelloWorldService对象从它的构造创建时,HelloWorld对象也被创建了。它是从StrutsHelloWorld或者SpringHelloWorld创建的。
在这个例子中,要创建一个HelloWorldService对象,HelloWorld对象(SpringHelloWorld或者StrutsHelloWorld)也同时被创建。目前来看,HelloWorld对象的创建是由HelloWorldService类本身来控制的。
这样,HelloWorldService对象对于SpringHelloWorld和StrutsHelloWorld这两个类是紧耦合的,它们之间的依赖关系是通过java代码绑定的。如果要调整这种依赖,只能够修改java代码并重新编译。这样,是非常不科学,且不方便的。
从这里,我们引入“IoC、控制反转”的概念。控制反转(Inversion of Control,缩写为IoC)的意思是说,我们将创建HelloWorld的控制转交给第三方。通常来说的第三方,就是指IoC容器。
IoC容器充当管理者的角色,创建HelloWorldService对象和HelloWorld对象,并通过setter方法传递HelloWorld对象到HelloWorldService。
这种,实现IoC的方式,叫做“依赖注入”。也就是说,将HelloWorldService依赖的HelloWorld对象,注入到HelloWorldService中。下面是一个示意图。


IoC和依赖注入举例

3、例子

在IDEA中创建一个空的Maven项目(需要选任何archetype),这里输入的groupid和artifactid如下。

<groupId>com.SpaceCat.SpringHelloWorld</groupId>
<artifactId>SpringHelloWorld</artifactId>
<version>1.0-SNAPSHOT</version>

然后,创建如下图的代码工程目录结构。


工程目录结构

其中,对应的代码如下。
HelloWorldProgram.java:

package com.SpaceCat.HelloWorld.spring;

import com.SpaceCat.HelloWorld.spring.helloworld.HelloWorld;
import com.SpaceCat.HelloWorld.spring.helloworld.HelloWorldService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

/**
 * Created by chengxia on 2019/4/30.
 */
public class HelloWorldProgram {
    public static void main(String[] args) {

        ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
        HelloWorldService service = (HelloWorldService) context.getBean("helloWorldService");
        HelloWorld hw= service.getHelloWorld();
        hw.sayHello();
    }

}

HelloWorld.java:

package com.SpaceCat.HelloWorld.spring.helloworld;

/**
 * Created by chengxia on 2019/4/30.
 */
public interface HelloWorld {
    public void sayHello();
}

HelloWorldService.java:

package com.SpaceCat.HelloWorld.spring.helloworld;

/**
 * Created by chengxia on 2019/4/30.
 */
public class HelloWorldService {
    private HelloWorld helloWorld;

    public HelloWorldService() {

    }

    public void setHelloWorld(HelloWorld helloWorld) {
        this.helloWorld = helloWorld;
    }

    public HelloWorld getHelloWorld() {
        return this.helloWorld;
    }
}

SpringHelloWorld.java:

package com.SpaceCat.HelloWorld.spring.helloworld.impl;

import com.SpaceCat.HelloWorld.spring.helloworld.HelloWorld;

/**
 * Created by chengxia on 2019/4/30.
 */
public class SpringHelloWorld implements HelloWorld {

    public void sayHello() {
        System.out.println("Spring Say Hello!!");
    }
}

StrutsHelloWorld.java:

package com.SpaceCat.HelloWorld.spring.helloworld.impl;

import com.SpaceCat.HelloWorld.spring.helloworld.HelloWorld;

/**
 * Created by chengxia on 2019/4/30.
 */
public class StrutsHelloWorld implements HelloWorld {
    public void sayHello() {
        System.out.println("Structs Say Hello!!");
    }
}

beans.xml:

<?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">
    <bean id="springHelloWorld" class="com.SpaceCat.HelloWorld.spring.helloworld.impl.SpringHelloWorld">
    </bean>
    <bean id="strutsHelloWorld" class="com.SpaceCat.HelloWorld.spring.helloworld.impl.StrutsHelloWorld">
    </bean>

    <bean id="helloWorldService"
          class="com.SpaceCat.HelloWorld.spring.helloworld.HelloWorldService">
        <property name="helloWorld" ref="springHelloWorld"/>
    </bean>
</beans>

ClassStructure.puml(该文件只是用于绘制这个工程的类结构图,这个是用的IDEA的PlantUML插件,和工程的运行没有关系,可以不用):

@startuml

Title "类图"
interface HelloWorld
HelloWorld <|.. SpringHelloWorld:实现
HelloWorld <|.. StrutsHelloWorld:实现
HelloWorldService ..> HelloWorld:依赖

@enduml

pom.xml:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.SpaceCat.SpringHelloWorld</groupId>
    <artifactId>SpringHelloWorld</artifactId>
    <version>1.0-SNAPSHOT</version>

    <dependencies>
        <!-- Spring Core -->
        <!-- http://mvnrepository.com/artifact/org.springframework/spring-core -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>4.1.4.RELEASE</version>
        </dependency>
        <!-- Spring Context -->
        <!-- http://mvnrepository.com/artifact/org.springframework/spring-context -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>4.1.4.RELEASE</version>
        </dependency>
    </dependencies>
</project>

完成之后,运行HelloWorldProgram.java文件,得到如下控制台输出,程序正确运行。

四月 30, 2019 7:37:02 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@6267c3bb: startup date [Tue Apr 30 19:37:02 CST 2019]; root of context hierarchy
四月 30, 2019 7:37:02 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
信息: Loading XML bean definitions from class path resource [beans.xml]
Spring Say Hello!!

Process finished with exit code 0

如下图中展示了IoC容器依赖注入的示意图。

IoC依赖注入示意图

而在上面结构中,由于引入了Spring框架,如果我们想要调整HelloWorldService对象对于SpringHelloWorld和StrutsHelloWorld的依赖关系,只需要修改beans.xml配置文件,无需修改java代码或者重新编译。
Spring框架提供的这种依赖注入(DI)特性极大地方便了对象依赖关系的管理,使大型Java项目开发更优雅,更灵活和更便于维护。
接下来,我们通过在上面的工程根目录下运行命令mvn dependency:tree来查看这个工程的用到的依赖和依赖之间的关系。如下。

$ mvn dependency:tree
[INFO] Scanning for projects...
[INFO] 
[INFO] -----------< com.SpaceCat.SpringHelloWorld:SpringHelloWorld >-----------
[INFO] Building SpringHelloWorld 1.0-SNAPSHOT
[INFO] --------------------------------[ jar ]---------------------------------
[INFO] 
[INFO] --- maven-dependency-plugin:2.8:tree (default-cli) @ SpringHelloWorld ---
[INFO] com.SpaceCat.SpringHelloWorld:SpringHelloWorld:jar:1.0-SNAPSHOT
[INFO] +- org.springframework:spring-core:jar:4.1.4.RELEASE:compile
[INFO] |  \- commons-logging:commons-logging:jar:1.2:compile
[INFO] \- org.springframework:spring-context:jar:4.1.4.RELEASE:compile
[INFO]    +- org.springframework:spring-aop:jar:4.1.4.RELEASE:compile
[INFO]    |  \- aopalliance:aopalliance:jar:1.0:compile
[INFO]    +- org.springframework:spring-beans:jar:4.1.4.RELEASE:compile
[INFO]    \- org.springframework:spring-expression:jar:4.1.4.RELEASE:compile
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time:  1.427 s
[INFO] Finished at: 2019-04-30T19:39:52+08:00
[INFO] ------------------------------------------------------------------------
$ 

参考资料

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,076评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,658评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,732评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,493评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,591评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,598评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,601评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,348评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,797评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,114评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,278评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,953评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,585评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,202评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,180评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,139评论 2 352

推荐阅读更多精彩内容