文档章节

Spring学习笔记(6)---------IoC的三种注入方式

Simon丶Ma
 Simon丶Ma
发布于 2016/04/14 14:57
字数 2484
阅读 16
收藏 0
点赞 1
评论 0
  1. 接口注入(不推荐
  2. 构造器注入(死的应用
  3. getter,setter方式注入(比较常用

Type1 接口注入

我们常常借助接口来将调用者与实现者分离。如:


  
  1. public class ClassA {  
  2.   private InterfaceB clzB;  
  3.   public init() {  
  4.   Ojbect obj =  
  5.   Class.forName(Config.BImplementation).newInstance();  
  6.   clzB = (InterfaceB)obj;  
  7.   }  
  8.   ……  
  9.   }  

  上面的代码中,ClassA依赖于InterfaceB的实现,如何获得InterfaceB实现类的实例?传统的方法是在代码中创建InterfaceB实现类的实例,并将起赋予clzB。

  而这样一来,ClassA在编译期即依赖于InterfaceB的实现。为了将调用者与实现者在编译期分离,于是有了上面的代码,我们根据预先在配置文件中设定的实现类的类名,动态加载实现类,并通过InterfaceB强制转型后为ClassA所用。
  这就是接口注入的一个最原始的雏形。
  而对于一个Type1型IOC容器而言,加载接口实现并创建其实例的工作由容器完成,如J2EE开发中常用的Context.lookup(ServletContext.getXXX),都是Type1型IOC的表现形式。
  Apache Avalon是一个典型的Type1型IOC容器。

Type2 构造子注入

构造子注入,即通过构造函数完成依赖关系的设定,如:


  
  1. public class DIByConstructor {  
  2.   private final DataSource dataSource;  
  3.   private final String message;  
  4.   public DIByConstructor(DataSource ds, String msg) {  
  5.   this.dataSource = ds;  
  6.   this.message = msg;  
  7.   }  
  8.   ……  
  9.   }  

  可以看到,在Type2类型的依赖注入机制中,依赖关系是通过类构造函数建立,容器通过调用类的构造方法,将其所需的依赖关系注入其中。
  PicoContainer(另一种实现了依赖注入模式的轻量级容器)首先实现了Type2类型的依赖注入模式。

Type3 设值注入

  在各种类型的依赖注入模式中,设值注入模式在实际开发中得到了最广泛的应用(其中很大一部分得力于Spring框架的影响)。
  在笔者看来,基于设置模式的依赖注入机制更加直观、也更加自然。Quick Start中的示例,就是典型的设置注入,即通过类的setter方法完成依赖关系的设置。

几种依赖注入模式的对比总结

  接口注入模式因为具备侵入性,它要求组件必须与特定的接口相关联,因此并不被看好,实际使用有限。
  Type2和Type3的依赖注入实现模式均具备无侵入性的特点。在笔者看来,这两种实现方式各有特点,也各具优势(一句经典废话?)。

    Type2 构造子注入的优势:

  1. “在构造期即创建一个完整、合法的对象”,对于这条Java设计原则,Type2无疑是最好的响应者。
  2. 避免了繁琐的setter方法的编写,所有依赖关系均在构造函数中设定,依赖关系集中呈现,更加易读。
  3. 由于没有setter方法,依赖关系在构造时由容器一次性设定,因此组件在被创建之后即处相对“不变”的稳定状态,无需担心上层代码在调用过程中执行setter方法对组件依赖关系产生破坏,特别是对于Singleton模式的组件而言,这可能对整个系统产生重大的影响。
  4. 同样,由于关联关系仅在构造函数中表达,只有组件创建者需要关心组件内部的依赖关系。对调用者而言,组件中的依赖关系处于黑盒之中。对上层屏蔽不必要的信息,也为系统的层次清晰性提供了保证。
  5. 通过构造子注入,意味着我们可以在构造函数中决定依赖关系的注入顺序,对于一个大量依赖外部服务的组件而言,依赖关系的获得顺序可能非常重要,比如某个依赖关系注入的先决条件是组件的DataSource及相关资源已经被设定。

    Type3 设值注入的优势:

  1. 对于习惯了传统JavaBean开发的程序员而言,通过setter方法设定依赖关系显得更加直观,更加自然。
  2. 如果依赖关系(或继承关系)较为复杂,那么Type2模式的构造函数也会相当庞大(我们需要在构造函数中设定所有依赖关系),此时Type3模式往往更为简洁。
  3. 对于某些第三方类库而言,可能要求我们的组件必须提供一个默认的构造函数(如Struts中的Action),此时Type2类型的依赖注入机制就体现出其局限性,难以完成我们期望的功能。

  可见,Type2和Type3模式各有千秋,而Spring、PicoContainer都对Type2和Type3类型的依赖注入机制提供了良好支持。这也就为我们提供了更多的选择余地。理论上,以Type2类型为主,辅之以Type3类型机制作为补充,可以达到最好的依赖注入效果,不过对于基于Spring Framework开发的应用而言,Type3使用更加广泛

借鉴http://blog.csdn.net/zhuangzhineng/article/details/3967234

 

 

一、构造器注入

     构造器注入顾名思义就是在程序组件中实现构造器,构造器可以是一个也可以是多个。废话不多说,直接上代码。

 

Java代码 复制代码   收藏代码
  1. package cn.glzaction.service.impl;  
  2.   
  3. import java.util.List;  
  4.   
  5. import cn.glzaction.service.interfaces.PersonDaoIF;  
  6. import cn.glzaction.service.interfaces.PersonServiceIF;  
  7.   
  8. public class PersonServiceBean implements PersonServiceIF{  
  9.   
  10. //自定义类  
  11.     private PersonDaoIF personDaoBean;  
  12. //String类型  
  13.     private String name;  
  14. //集合类型    
  15.     private List list;  
  16.       
  17. //构造器     
  18.     public PersonServiceBean(PersonDaoBean personDaoBean,String name,List list){  
  19.         this.personDaoBean = personDaoBean;  
  20.         this.name = name;  
  21.         this.list = list;  
  22.     }  
  23. //方法,用于显示  
  24.     public void display(){  
  25.         personDaoBean.add();  
  26.         System.out.println(name);  
  27.         System.out.println(list);  
  28.     }  
  29. }  
<span style="font-size:18px;">package cn.glzaction.service.impl;

import java.util.List;

import cn.glzaction.service.interfaces.PersonDaoIF;
import cn.glzaction.service.interfaces.PersonServiceIF;

public class PersonServiceBean implements PersonServiceIF{

//自定义类
	private PersonDaoIF personDaoBean;
//String类型
	private String name;
//集合类型	
	private List list;
	
//构造器	
	public PersonServiceBean(PersonDaoBean personDaoBean,String name,List list){
		this.personDaoBean = personDaoBean;
		this.name = name;
		this.list = list;
	}
//方法,用于显示
	public void display(){
		personDaoBean.add();
		System.out.println(name);
		System.out.println(list);
	}
}</span>

 

 

 

上面的代码中构造器要注入三个参数,同时这三个参数也是三种不同的类型,自定义类、String类型,集合类型,其中自定义类PersonDaoBean 具体的实现我们这里就不累述了,因为这不是重点。下面我们再来看一下在Spring的配置文件中如何来配置。

 

Xml代码 复制代码   收藏代码
  1. <bean id="personDao" class="cn.glzaction.service.impl.PersonDaoBean"></bean >  
  2.   
  3. <!--构造器方式注入-->  
  4. <bean id="personService" class="cn.glzaction.service.impl.PersonServiceBean">  
  5.     <constructor-arg index="0" type="cn.glzaction.service.impl.PersonDaoBean" ref="personDao"/>  
  6.     <constructor-arg index="1" type="java.lang.String" value="glzaction"/>  
  7.     <constructor-arg index="2" type="java.util.List">  
  8.         <list>  
  9.             <value>list1</value>  
  10.             <value>list2</value>  
  11.             <value>list3</value>  
  12.         </list>  
  13.     </constructor-arg>  
  14. </bean>  
<span style="font-size:18px;"><bean id="personDao" class="cn.glzaction.service.impl.PersonDaoBean"></bean >

<!--构造器方式注入-->
<bean id="personService" class="cn.glzaction.service.impl.PersonServiceBean">
	<constructor-arg index="0" type="cn.glzaction.service.impl.PersonDaoBean" ref="personDao"/>
	<constructor-arg index="1" type="java.lang.String" value="glzaction"/>
	<constructor-arg index="2" type="java.util.List">
		<list>
			<value>list1</value>
			<value>list2</value>
			<value>list3</value>
		</list>
	</constructor-arg>
</bean></span>

 

 

      java代码如何实现构造器我想大家一定都很清楚,否则的话就有点讲不过去了,呵呵,所以这里我也就不加以说明了,主要说一下上面的xml配置。id为“personDao”的bean是PersonServiceBean的私有属性,它的注入是采用无参构造器的注入方式注入的,这也不详细说明。主要说一下id为“personService”类的构造器注入。<coustructor-arg>是构造器标签元素,通过设定它的属性可以往构造器传递参数,index属性值表示要设定的参数在构造器形参中的索引顺序,例如上面的配置,list是第三个参数,所以它对应的索引为2,index是可选属性,所谓可选并不是说在任何情况下都可以不使用,要视具体情况而定,type为参数的类型,这个也是可选参数。还有两个很重要的属性就是ref和value,如果注入的是bean,就要使用ref,ref的值就是对应的bean。如果注入的是基本类型或者string类型就用value,直接将对应的值填入即可。

 

二、设值注入(setter方式注入)

设值注入就是通过setXxxx方法将bean注入到组件中,自定义类如下

 

Java代码 复制代码   收藏代码
  1. package cn.glzaction.service.impl;  
  2.   
  3. import cn.glzaction.service.interfaces.PersonDaoIF;  
  4. import java.util.*;  
  5.   
  6. public class PersonDaoBean implements PersonDaoIF {  
  7.   
  8.     private String name;  
  9.       
  10.     private Integer id;  
  11.       
  12.     private List list;  
  13.       
  14.     private Map map;  
  15.       
  16.     public void setName(String name) {  
  17.         this.name = name;  
  18.     }  
  19.     public void setId(Integer id) {  
  20.         this.id = id;  
  21.     }  
  22.     public void setList(List list) {  
  23.         this.list = list;  
  24.     }  
  25.     public void setMap(Map map) {  
  26.         this.map = map;  
  27.     }  
  28.   
  29.     @Override  
  30.     public void add() {  
  31.         // TODO Auto-generated method stub  
  32.         System.out.println(map);  
  33.         System.out.println(list);  
  34.         System.out.println(id);  
  35.         System.out.println(name);  
  36.     }  
  37. }  
<span style="font-size:18px;">package cn.glzaction.service.impl;

import cn.glzaction.service.interfaces.PersonDaoIF;
import java.util.*;

public class PersonDaoBean implements PersonDaoIF {

	private String name;
	
	private Integer id;
	
	private List list;
	
	private Map map;
	
	public void setName(String name) {
		this.name = name;
	}
	public void setId(Integer id) {
		this.id = id;
	}
	public void setList(List list) {
		this.list = list;
	}
	public void setMap(Map map) {
		this.map = map;
	}

	@Override
	public void add() {
		// TODO Auto-generated method stub
		System.out.println(map);
		System.out.println(list);
		System.out.println(id);
		System.out.println(name);
	}
}</span>

 

 

 

采用设置注入只要有setter方法即可,但是有时由于编程习惯也会讲getter方法引进,但是要清楚:设值注入与getter方法无关。这里还有一点需要注意,那就是能使用基本类型,如果非要使用基本类型的话就要使用其对应的包装类型,如上面使用的是Integer而不是int。Spring的xml配置如下:

 

Xml代码 复制代码   收藏代码
  1. <bean id="personDao" class="cn.glzaction.service.impl.PersonDaoBean">  
  2.     <property name="name" type="java.lang.String" value="glzaction"/>  
  3.     <property name="id" type="java.lang.Integer" value="1"/>  
  4.     <property name="list" type="java.util.List">  
  5.         <list>  
  6.             <value>list1</value>  
  7.             <value>list2</value>  
  8.             <value>list3</value>  
  9.         </list>  
  10.     </property>  
  11.     <property name="map" type="java.util.Map">  
  12.         <map>  
  13.             <entry key="key1" value="value1"></entry>  
  14.             <entry key="key2" value="value2"></entry>  
  15.         </map>  
  16.     </property>  
  17. </bean>  
<span style="font-size:18px;"><bean id="personDao" class="cn.glzaction.service.impl.PersonDaoBean">
	<property name="name" type="java.lang.String" value="glzaction"/>
	<property name="id" type="java.lang.Integer" value="1"/>
	<property name="list" type="java.util.List">
		<list>
			<value>list1</value>
			<value>list2</value>
			<value>list3</value>
		</list>
	</property>
	<property name="map" type="java.util.Map">
		<map>
			<entry key="key1" value="value1"></entry>
			<entry key="key2" value="value2"></entry>
		</map>
	</property>
</bean></span>

 

 设值注入采用的是<property>标签元素,其中的name属性对应的是要注入的变量名,type属性值对应的该变量的类型,可以是自定义类或者包装类型。value属性对应的是相应的值,还有一个ref属性,该属性值对应的是bean。

 

 

 

 

© 著作权归作者所有

共有 人打赏支持
Simon丶Ma
粉丝 4
博文 134
码字总数 299850
作品 0
深圳
程序员
IOC/AOP工具 - jBeanBox

jBeanBox是一个微形但功能较齐全的IOC/AOP工具适用于JAVA7+,利用了Java的初始化块实现的Java配置代替XML。jBeanBox采用Apache License 2.0开源协议。 其他一些IOC/AOP框架的问题: 1)Sprin...

yong9981 ⋅ 2016/07/25 ⋅ 14

依赖注入/控制反转学习笔记

最近在学习js的开发框架比如AngularJS、Meteor、Backbone等,这些框架的一个特点就是依赖注入,因为一直对这个概念云山雾罩,所以特意花时间学习了一下,把学习笔记写在这里。 ------------...

路过暴风 ⋅ 2014/01/14 ⋅ 0

Spring IOC知识点一网打尽!

前言 只有光头才能变强 回顾前面: 给女朋友讲解什么是代理模式 包装模式就是这么简单啦 单例模式你会几种写法? 工厂模式理解了没有? 在刷Spring书籍的时候花了点时间去学习了单例模式和工...

Java3y ⋅ 05/22 ⋅ 0

Spring之IoC(控制反转)和DI(依赖注入)

1.IoC的概念 IoC:通过容器去控制业务对象之间的依赖关系。控制权由应用代码中转到了外部容器,控制权的转移就是反转。控制权转移的意义是降低了类之间的耦合度。 Spring中将IoC容器管理的对...

Java攻城玩家 ⋅ 05/28 ⋅ 0

Spring.NET学习笔记——目录(原)

目录 前言 Spring.NET学习笔记——前言 第一阶段:控制反转与依赖注入IoC&DI Spring.NET学习笔记1——控制反转(基础篇) Level 200 Spring.NET学习笔记2——环境搭建(基础篇) Level 200 Sprin...

长平狐 ⋅ 2012/06/11 ⋅ 1

Spring源码解析系列之IOC容器(一)

前言 实际上我所有的博客都是原来对原来印象笔记里笔记内容的加工,关于Spring源码自己已经解析了很多遍,但是时间长总是忘记,写一篇博客权当加强记忆,也算再次学习下大师们的设计思想,思...

后厂村老司机 ⋅ 06/02 ⋅ 0

SpringBoot的幕后推手...

一、背景   这两年随着微服务的盛行,SpringBoot框架水到渠成的得到了高曝光,作为程序猿的我们,现在要是不知道一点SpringBoot相关的东西,貌似出去找工作都会被深深地鄙视,不过在我们开...

Java工程师-Distance ⋅ 06/01 ⋅ 0

让Spring Security 来保护你的Spring Boot项目吧

参考资料: 书籍:Spring实战(第4版) 第9章和第14章 Spring Security 参考手册 初识 Spring Security 程序猿DD的Spring Security学习笔记 Spring Security 简介 Spring Security是一个能够为...

潇潇漓燃 ⋅ 05/19 ⋅ 0

阿里老司机带你使用Spring框架快速搭建Web工程项目

摘要:Spring 框架是一个开源的 Java 平台,它为容易而快速的开发出耐用的 Java 应用程序提供了全面的基础设施。借助于Spring框架可以快速搭建Web工程项目,本文中阿里巴巴高级开发工程师嵛山...

萌萌怪兽 ⋅ 05/15 ⋅ 0

Spring IoC & DI

IOC诞生的历史 在没有IoC时,关联不同模块是通过类实例实现的,代码可能是这样子的: 当YourServiceImpl的接口不变时,只需要根据业务需要更换不同的YourService实现类即可。一旦更换实现类时...

classfly ⋅ 06/18 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

浅谈springboot Web模式下的线程安全问题

我们在@RestController下,一般都是@AutoWired一些Service,由于这些Service都是单例,所以并不存在线程安全问题。 由于Controller本身是单例模式 (非线程安全的), 这意味着每个request过来,...

算法之名 ⋅ 今天 ⋅ 0

知乎Java数据结构

作者:匿名用户 链接:https://www.zhihu.com/question/35947829/answer/66113038 来源:知乎 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。 感觉知乎上嘲讽题主简...

颖伙虫 ⋅ 今天 ⋅ 0

Confluence 6 恢复一个站点有关使用站点导出为备份的说明

推荐使用生产备份策略。我们推荐你针对你的生产环境中使用的 Confluence 参考 Production Backup Strategy 页面中的内容进行备份和恢复(这个需要你备份你的数据库和 home 目录)。XML 导出备...

honeymose ⋅ 今天 ⋅ 0

JavaScript零基础入门——(九)JavaScript的函数

JavaScript零基础入门——(九)JavaScript的函数 欢迎回到我们的JavaScript零基础入门,上一节课我们了解了有关JS中数组的相关知识点,不知道大家有没有自己去敲一敲,消化一下?这一节课,...

JandenMa ⋅ 今天 ⋅ 0

火狐浏览器各版本下载及插件httprequest

各版本下载地址:http://ftp.mozilla.org/pub/mozilla.org//firefox/releases/ httprequest插件截至57版本可用

xiaoge2016 ⋅ 今天 ⋅ 0

Docker系列教程28-实战:使用Docker Compose运行ELK

原文:http://www.itmuch.com/docker/28-docker-compose-in-action-elk/,转载请说明出处。 ElasticSearch【存储】 Logtash【日志聚合器】 Kibana【界面】 答案: version: '2'services: ...

周立_ITMuch ⋅ 今天 ⋅ 0

使用快嘉sdkg极速搭建接口模拟系统

在具体项目研发过程中,一旦前后端双方约定好接口,前端和app同事就会希望后台同事可以尽快提供可供对接的接口方便调试,而对后台同事来说定好接口还仅是个开始、设计流程,实现业务逻辑,编...

fastjrun ⋅ 今天 ⋅ 0

PXE/KickStart 无人值守安装

导言 作为中小公司的运维,经常会遇到一些机械式的重复工作,例如:有时公司同时上线几十甚至上百台服务器,而且需要我们在短时间内完成系统安装。 常规的办法有什么? 光盘安装系统 ===> 一...

kangvcar ⋅ 昨天 ⋅ 0

使用Puppeteer撸一个爬虫

Puppeteer是什么 puppeteer是谷歌chrome团队官方开发的一个无界面(Headless)chrome工具。Chrome Headless将成为web应用自动化测试的行业标杆。所以我们很有必要来了解一下它。所谓的无头浏...

小草先森 ⋅ 昨天 ⋅ 0

Java Done Right

* 表示难度较大或理论性较强。 ** 表示难度更大或理论性更强。 【Java语言本身】 基础语法,面向对象,顺序编程,并发编程,网络编程,泛型,注解,lambda(Java8),module(Java9),var(...

风华神使 ⋅ 昨天 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部