Spring之IOC概念、Bean对象创建及DI注入的三种方式
Spring的IOC(inverse of control )如何理解
首先我们先比对Java中两类创建对象的方式。
方式1:通过new关键字创建实例对象
private IAccountDao accountDao= new AccountDaoImpl();
方式2:通过工厂获取实例对象
private IAccountDao accountDao = BeanFactory.getBean("accountDao");
通过方式1创建对象时,我们是主动寻找所要创建的对象的,应用直接和资源联系,存在明显的依赖关系,无论是应用还是资源都很难独立。如图所示。
通过方式2创建对象时,应用与资源之间不再直接联系,而是通过工厂获取资源,由工厂与资源取得联系,并把对应的资源提供给应用。
对于方式1来说,创建什么类型的对象是完全由自己控制的,想new什么类型就new什么类型。而方式2则把创建对象的权利交给了BeanFactory这个类,通过固定的“名称”来获取想要的bean对象,而所得的对象是否是所需的或可用的则无法得知,因为获得的类对象是由"accountDao"
所对应的全限定类名所决定的,无法自主控制。这种创建对象的控制权的转移,我们称之为控制反转(Inverse of Control,IOC),此种方式可以降低程序间的依赖关系,削减耦合。
Spring对Bean的管理细节
1. 创建Bean对象的三种方式
使用默认构造函数创建
在spring的配置文件中使用bean标签,配以id和class属性之后,且没有其他属性和标签时。
采用的就是默认构造函数创建bean对象,此时如果类中没有默认构造函数,则无法创建。 创建bean对象的所属的类
public class AccountServiceImpl implements IAccountService {public AccountServiceImpl(){System.out.println("对象创建了");} }
bean
标签内容如下:<bean id="accountService" class="com.xmy.service.impl.AccountServiceImpl"></bean>
使用普通工厂中的方法创建对象(使用某个类中的方法创建对象,并存入spring容器)
工厂类如下:
public class InstanceFactory {public IAccountService getAccountService(){return new AccountServiceImpl();} }
bean
标签内容如下:<bean id="instanceFactory" class="com.xmy.factory.InstanceFactory"></bean> <bean id="accountService" factory-bean="instanceFactory" factory-method="getAccountService"></bean>
这种方法先通过第一行的bean标签获取InstanceFactory工厂类的对象,根据第二个bean标签中的
factory-bean
的值找到对应的工厂对象,并调用工厂对象中factory-method
所对应的方法来创建bean对象。使用工厂类的静态方法创建对象(使用某个类中的静态方法创建对象,并存入spring容器)
工厂类如下:
public class StaticFactory {public static IAccountService getAccountService(){return new AccountServiceImpl();} }
bean
标签内容如下:<bean id="accountService" class="com.xmy.factory.StaticFactory" factory-method="getAccountService"></bean>
由于采用了工厂类的静态方法创建对象,所以,无需实例化工厂类对象,只需要直接调用工厂类的静态方法创建bean对象。
2. bean对象的作用范围
bean标签的scope属性:用于指定bean的作用范围,取值包括:
- singleton:单例的
- prototype:多例的
- request:作用于web应用的请求范围
- session:作用于web应用的会话范围
- global-session:作用于集群环境(全局)的会话范围,当不是集群环境时,它就是session
3.bean对象的生命周期
单例对象
出生:当容器创建时,对象出生
存活:只要容器还在,对象一直存在
死亡:容器销毁,对象消亡
多例对象
出生:当我们使用对象时,spring框架为我们创建
存活:对象只要是在使用过程中就一直存活
死亡:当对象长时间不用,且没有别的对象引用时,由Java垃圾回收器回收
Spring中的DI(依赖注入)
当一个类需要用到其他类的对象时,通常会在调用类中直接new一个所需的对象,这样就形成了类之间的依赖关系,增加了程序的耦合,而为了尽可能的避免这类情况,降低程序的耦合,Spring框架采用了“依赖注入”来解决,在当前类需要用到其他类的对象,由spring为我们提供,我们只需要在配置文件中说明。
所谓的依赖注入简单来说就是给一个对象传入实例变量,这个实例变量可以称为“依赖”。当我们使用这些实例变量(依赖)时,使用的是传递进来的对象,而不是由我们自己创建的对象。所以不会形成类之间的依赖关系,实现解耦。
依赖注入的数据分为三种:1. 基本类型和String 2.其他bean类型(在配置文件中或者注解配置过的bean) 3.复杂类型/集合类型
依赖注入的方式有三种:1.使用构造函数注入,2.使用Set方法注入,3.使用注解注入
1.使用构造函数注入
使用标签:constructor-arg
标签出现的位置:bean
标签的内部
标签中的属性:
type
:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些参数的类型
index
:用于指定注入数据的赋值对象于构造函数中的索引位置。索引的位置是从0开始
name
:用于指定给构造函数中指定名称的参数赋值(常用)
————以上三个用于指定给构造函数中哪个参数赋值————
value
:用于提供基本类型和String类型的数据
ref
:用于指定其他的bean类型数据。它指的就是在spring的IOC核心容器中出现过的bean对象
优势:在获取bean对象时,注入数据是必须的操作,否则对象无法创建成功。对象初始化后即可 直接使用。
缺点:在创建对象时,如果构造函数需求了某些参数对象,即使用不到,也必须提供。
创建业务层
public class AccountServiceImpl implements IAccountService {private String name;private Integer age;private Date birthday;public AccountServiceImpl(String name, Integer age, Date birthday) {this.name = name;this.age = age;this.birthday = birthday;}public void saveAccount() {System.out.println("service中的saveAccount方法执行了..."+name+","+age+","+birthday+",");}}
创建bean标签
<bean id="accountService" class="com.xmy.service.impl.AccountServiceImpl"><constructor-arg name="name" value="test"></constructor-arg><constructor-arg name="age" value="18"></constructor-arg><constructor-arg name="birthday" ref="now"></constructor-arg>
</bean><!--配置一个日期对象--><bean id="now" class="java.util.Date"></bean>
创建表现层
public class Client {public static void main(String[] args) {//1.获取核心容器对象ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");//2.根据id获取Bean对象IAccountService as = (IAccountService) ac.getBean("accountService");as.saveAccount();}}
运行结果如图
2.使用Set方法注入
涉及的标签:property
出现的位置:bean标签的内部
标签的属性:
name:用于指定注入时所调用的set方法名称
value:用于提供基本类型和String类型的数据
ref:用于指定其他的bean类型数据。它指的就是在spring的IOC核心容器中出现过的bean对象
优势:可以使用默认构造函数创建对象,并根据实际需要,在对象声明周期内随时动态的改变注入 的依赖。
缺点:对象被初始化之后,无法立即使用,需要注入依赖后才可以。
创建业务层
public class AccountServiceImpl2 implements IAccountService {private String name;private Integer age;private Date birthday;public void setName(String name) {this.name = name;}public void setAge(Integer age) {this.age = age;}public void setBirthday(Date birthday) {this.birthday = birthday;}public void saveAccount() {System.out.println("service中的saveAccount方法执行了..."+name+","+age+","+birthday+",");}}
创建bean标签
<bean id="accountService2" class="com.xmy.service.impl.AccountServiceImpl2"><property name="username" value="test"></property><property name="age" value="27"></property>
</bean>
创建表现层
public class Client {public static void main(String[] args) {//1.获取核心容器对象ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");//2.根据id获取Bean对象IAccountService as = (IAccountService) ac.getBean("accountService2");as.saveAccount();}
}
运行结果如图
可以看到,由于没有创建name
属性为birthday
的property
标签,因此没有注入Date
类型的对象,所以打印结果为null。
下面介绍一下对于复杂类型/集合类型的注入
创建业务层
public class AccountServiceImpl3 implements IAccountService {private String[] myStr;private List<String> myList;private Set<String> mySet;private Map<String,String> myMap;private Properties myProps;public void setMyStr(String[] myStr) {this.myStr = myStr;}public void setMyList(List<String> myList) {this.myList = myList;}public void setMySet(Set<String> mySet) {this.mySet = mySet;}public void setMyMap(Map<String, String> myMap) {this.myMap = myMap;}public void setMyPro(Properties myProps) {this.myProps = myProps;}public void saveAccount() {System.out.println(Arrays.toString(myStr));System.out.println(myList);System.out.println(mySet);System.out.println(myMap);System.out.println(myProps);}
}
创建bean标签
<bean id="accountService3" class="com.xmy.service.impl.AccountServiceImpl3"><property name="myList"><array><value>AAA</value><value>bbb</value><value>CCC</value></array></property><property name="mySet"><list><value>AAA</value><value>bbb</value><value>CCC</value></list></property><property name="myStr"><list><value>AAA</value><value>bbb</value><value>CCC</value></list></property><property name="myPro"><map><entry key="testC" value="ccc"></entry><entry key="testD" value="ddd"></entry></map></property><property name="myMap"><props><prop key="testA">aaa</prop><prop key="testB">bbb</prop></props></property>
</bean>
创建表现层
public class Client {public static void main(String[] args) {//1.获取核心容器对象ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");//2.根据id获取Bean对象IAccountService as = (IAccountService) ac.getBean("accountService3");as.saveAccount();}
}
运行结果如图
虽然在bean
标签中,每一个属性标签都没有对应上,但是仍然正常的运行并打印出了结果。这是因为对于结构相同的集合类型,标签可以互换。
用于给List结构集合注入的标签:list array set
用于给Map结构集合注入的标签:map props
3.使用@Autowired注解注入
其作用与在xml配置文件中的标签中写一个标签的功能是一样的。@Autowired注解
可以对类的成员变量、方法及构造函数进行标注,自动按照类型注入。只要容器中有唯一的一个bean对象类型和要注入的变量类型匹配,就可以注入成功。
下面用具体案例说明
定义一个账户持久层的接口IAccountDao,其中定义一个saveAccount方法
/*** 账户的持久层接口* @author xmy*/
public interface IAccountDao {/*** 模拟保存账户*/void saveAccount();
}
定义一个账户业务层接口IAccountService,其中同样定义一个saveAccount方法
/*** 账户业务层的接口* @author xmy*/
public interface IAccountService {/*** 模拟保存账户*/void saveAccount();
}
AccountDaoImpl.java
@Repository("accountDao")
public class AccountDaoImpl implements IAccountDao {public void saveAccount() {System.out.println("保存了账户");}
}
定义一个IAccountDao接口的实现类,并实现saveAccount方法,该bean在IOC容器中的标识符为accountDao.
AccountServiceImpl.java
@Service("accountService")
public class AccountServiceImpl implements IAccountService {@Autowiredprivate IAccountDao accountDao = null;public void saveAccount() {accountDao.saveAccount();}
}
定义一个IAccountService接口的实现类,并实现saveAccount方法,该bean在IOC容器中的标识符为accountService.
bean.xml配置
<context:component-scan base-package="com.xmy"></context:component-scan>
测试代码:
public class Client {/*** @param args*/public static void main(String[] args) {//1.获取核心容器对象ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");//2.根据id获取Bean对象IAccountService as = (IAccountService) ac.getBean("accountService");as.saveAccount();}
}
运行结果为:
保存了账户
为了便于理解,这里通过一张图来解释@Autowired的执行过程
如图所示,当ioc容器扫描到@Autowired注解后,就会容器内自动查找与注入数据类型相匹配的bean,装配给该参数。如果没有找到相匹配的bean,则会报错。而如果IOC容器中有多个bean与注入数据类型相匹配,那么IOC容器会根据注入数据类型的名称作为key来找对应的bean,并装配给该参数。
IAccountDaoImpl2.java
@Repository("accountDao2")
public class AccountDaoImpl2 implements IAccountDao {public void saveAccount() {System.out.println("保存了账户22222");}
}
再定义一个IAccountDao接口的实现类IAccountDaoImpl2,并实现saveAccount方法,该bean在IOC容器中的标识符为accountDao2.
如果IOC容器中有多个bean与注入数据类型相匹配,那么IOC容器会根据注入数据类型的名称作为key来找对应的bean,并装配给该参数。@Autowired装配过程如图所示。
Spring之IOC概念、Bean对象创建及DI注入的三种方式相关推荐
- Spring属性注入的三种方式(超详细)
属性注入的三种方式 使用set方法进行注入 使用有参构造函数进行注入 使用p名称空间注入 首先了解下面两个名词的含义: IOC:控制反转(Inversion of Control,缩写为IoC),是面 ...
- 05.bean依赖注入的三种方式
05.bean依赖注入的三种方式 1.概述 依赖注入 DI(Dependency Injection):它是 Spring 框架核心 IOC 的具体实现. 在编写程序时,通过控制反转,把对象的创建交给 ...
- spring依赖注入的三种方式以及优缺点
spring依赖注入的三种方式以及优缺点 一.依赖注入的三种方式 1.通过构造器注入.(spring4.3之后,推荐使用) 2.通过setter注入.(spring4.3之前,推荐使用) 3通过fil ...
- Spring注入的三种方式
Spring实例注入的三种方式: 1.属性注入,即使用注解注入. 2.set方法注入. 3.构造方法注入. 1.属性注入 使用@Autowired.@Resource或@Inject注解注入. 1.1 ...
- 【玩转SQLite系列】(二)SQLite创建和打开数据库的三种方式
转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/53234396 本文出自[DylanAndroid的博客] [玩转SQLite系 ...
- IoC(控制反转)的主要组件和注入的两种方式
一.IoC的主要组件: (1).Spring框架的两个最基本和最重要的包是org.springframework.beans.factory(该包中的主要接口是BeanFactory)和org.spr ...
- 依赖注入的三种方式_Spring IoC是如何进行依赖注入的
依赖注入(DI) DI(Dependency Injection),Spring IoC 不是一种技术,而是一种思想,通过这种思想,能够指导我们设计出松耦合的程序代码.而Spring IoC这个思想的 ...
- Spring系列之依赖注入的三种方式
目录 一.依赖注入方式 1.使用属性的setXXX方法注入 2.构造函数注入 (1)按类型匹配入参type (2)按索引匹配入参index (3)联合使用类型和索引匹配入参[type和index一起使 ...
- Spring详解-------依赖注入的三种方式实例详解
目录 1.什么是依赖注入 1.1类的关系 1.1.1 依赖关系(Dependency) 1.1.2 聚合(Aggregation) 1.2关系强度 2 为什么使用依赖注入 2.1开闭原则 2.1.1 ...
- Spring注解依赖注入的三种方式的优缺点以及优先选择
当我们在使用依赖注入的时候,通常有三种方式: 1.通过构造器来注入: 2.通过setter方法来注入: 3.通过filed变量来注入: 那么他们有什么区别吗?应该选择哪种方式更好? 代码示例: Con ...
最新文章
- Developer Express XtraGrid使用技巧
- iOS面试题总结 二
- .prop()与.attr()
- 5分钟实现Android中更换头像功能
- python入门作业编程题-Python语言编写有趣练习题!
- 牛客多校6 - Binary Vector(组合数学+推公式)
- 如何学习c语言 零基础20天学会C语言
- dataframe for循环 筛选_Python循环12种超强写法,又快又省内存
- 操作系统之进程管理:5、处理机调度
- 静态的html页面想要设置使用浏览器缓存
- python中什么具有去重功能_python中去重的方法
- 纯新手DSP编程--5.30--任务的通信和同步
- scrapy vs requests+beautifulsoup
- 如何c51和mdk共存兼容_解决KeilMDK和KeilC51的兼容性——如何只用一个keil软件既可编译51核又可编译ARM核...
- Display debug(Blue Screen , fliker)
- 让AI拥有好奇心,它就可以一直看电视了
- windows下命令行格式化U盘
- anaconda无法安装最新版pip
- String.length() 与 String.getBytes().length
- 前端的扁平化是什么意思