Spring中的实例生成方式及其生命周期
三种实例化bean的方式
1.使用类构造器实例化
<!-- 使用类构造器实例化,class属性表示要使用的类的全限定名 -->
<bean id="userDao1"
class="com.winner.dao.daoImpl.UserDaoImpl">
</bean>
使用反射利用类的无参构造器生成实例。
2.使用静态工厂方法实例化
--配置文件:
<!-- 使用静态工厂方法实例化
class属性表示工厂类的全限定名
factory-method属性表示这个工厂类中用于创建实例的静态方法名(必须是static的)
-->
<bean id="userDao2"
class="com.winner.dao.StaticDaoFactory"
factory-method="createUserDaoInstance">
</bean>
--工厂类:
public class StaticDaoFactory {
// 方法必须声明为static的
public static Object createUserDaoInstance() {
System.out.println("StaticDaoFactory.createUserDaoInstance()");
return new UserDaoImpl();
}
}
3.使用实例化的工厂对象中的方法实例化,首先要构造出工厂实例,然后在生成实例
--配置文件:
<!-- 使用实例化工厂对象中的方法实例化
一、定义工厂bean
二、定义这个bean是由工厂方法创建的,其中:
factory-bean属性表示工厂bean的名称(id或name)
factory-method属性表示这个工厂类中用于创建实例的方法名(不能是static的)
-->
<bean id="simpleDaoFactory"
class="com.winner.dao.SimpleDaoFactory"></bean>
<bean id="userDao3"
factory-bean="simpleDaoFactory"
factory-method="createUserDaoInstance">
</bean>
--工厂类:
public class SimpleDaoFactory {
// 方法不能声明为static的
public static Object createUserDaoInstance() {
System.out.println("SimpleDaoFactory.createUserDaoInstance()");
return new UserDaoImpl();
}
}
我们一般写代码的时候都会分层,显示层,业务层,数据访问层,显示层调用业务层,业务层调用数据访问层。
数据访问层Dao只需要一个实例,业务层Service只需要一个实例,但是action每一次访问都会生成一个实例。
------------
Bean的作用域
在Spring IoC容器中定义的bean默认只有一个实例(单例的),且默认情况下会在容器启动时初始化bean(饿汉式),但我们可以指定Bean节点的lazy-init="true"来延迟初始化bean(懒汉式),这时候,只有第一次获取bean会才初始化bean。如:
--创建类:
public class User {
public User() {
System.out.println("初始化了User...");
}
}
--配置文件:
<!--
scope属性:用于指定bean的生命周期 singleton:单例的,每次ac.getBean()返回的都是同一个实例.
prototype:多例的,第次ac.getBean()返回的都是一个新的实例. 默认为singleton。
lazy-init属性:用于指定在什么时候初始化单例的对象
false:表示在创建ApplicationContext时就初始化本单例的对象。
true:表示在第一次调用getBean()获取本对象时才初始化。 默认为default.
-->
<bean id="user1" class="com.winner.spring.User"/>
或是:
<bean id="user1" class="com.winner.spring.User" scope="singleton"/>
或是:
<bean id="user1" class="com.winner.spring.User" scope="singleton" lazy-init="default"/>
或是
<bean id="user1" class="com.winner.spring.User" scope="singleton" lazy-init="false"/>
以上几种配置,都是在ApplicationContext容器创建时就生成了User对象。
<bean id="user1" class="com.winner.spring.User" scope="singleton" lazy-init="true"/>
当lazy-init取值true时,容器创建不会创建User对象,只有在getBean()时才会创建对象,即延迟加载
对所有bean都应用延迟初始化:
方法是在根节点beans中设置属性default-lazy-init="true",如下所示:
<beans default-lazy-init="true" ...>
prototype(原型,表示每次获取的都是新对象)
每次从容器获取bean都是新的对象。
<bean id="user1" class="com.winner.spring.User" scope="prototype"/>
getBean()时生成实例。
指定Bean的初始化方法和销毁方法
--创建类:
public class UserDao {
private DataSource dataSource;
public DataSource getDataSource() {
return dataSource;
}
public void setDataSource(DataSource dataSource) {
this.dataSource = dataSource;
}
/**
* 初始化方法
*/
public void init() {
System.out.println("UserDao.init() 初始化方法");
}
/**
* 销毁的方法
*/
public void destroy() {
System.out.println("UserDao.destroy() 销毁的方法");
}
}
--配置:
<!-- 单例的对象,配置了初始化方法与销毁方法 -->
<bean id="userDao" class="com.winner.spring.UserDao"
init-method="init" destroy-method="destroy"
/>
--测试代码:
// 在ApplicationContext接口中没有定义close()方法,要想调用,就得先转为子类类型才行。
// 一定要关闭ApplicationContext,给bean配置的销毁方法才会被调用。
// 在单例时,配置的初始化与销毁方法都会被调用。
// 在多例时,只有配置的初始化才会被调用。
@Test
public void test() throws Exception {
ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml", getClass());
UserDao userDao = (UserDao) ac.getBean("userDao");
System.out.println(userDao); ac.close();
}
--注意:
---如果scope属性为prototype则不受spring缓存的控制,destory方法也将不会执行(scope调为singleton时才会有效)。
---要调用ApplicationContext的close()方法才会执行destory方法(在ApplicationContext接口中没有close()方法,需要强转为具体的实现类才可以调用)
Spring中的实例生成方式及其生命周期的更多相关文章
- Spring中与bean有关的生命周期
前言 记得以前的时候,每次提起Spring中的bean相关的生命周期时,内心都无比的恐惧,因为好像有很多,自己又理不清楚,然后看网上的帖子,好像都是那么一套,什么beanFactory啊,aware接 ...
- 详解Spring中Bean的作用域与生命周期
摘要:在利用Spring进行IOC配置时,关于bean的配置和使用一直都是比较重要的一部分,同时如何合理的使用和创建bean对象,也是小伙伴们在学习和使用Spring时需要注意的部分,所以这一篇文章我 ...
- Spring中Bean的作用域、生命周期
Bean的作用域.生命周期 Bean的作用域 Spring 3中为Bean定义了5中作用域,分别为singleton(单例).protot ...
- Spring中bean的作用域与生命周期
在 Spring 中,那些组成应用程序的主体及由 Spring IOC 容器所管理的对象,被称之为 bean.简单地讲,bean 就是由 IOC 容器初始化.装配及管理的对象,除此之外,bean 就与 ...
- Spring中Bean的作用域和生命周期
作用域的种类 Spring 容器在初始化一个 Bean 的实例时,同时会指定该实例的作用域.Spring3 为 Bean 定义了五种作用域,具体如下. 1)singleton 单例模式,使用 sing ...
- 阶段3 2.Spring_03.Spring的 IOC 和 DI_8 spring中bean的细节之生命周期
区分单例还是多例对象 单例的几个状态 初始化方法和销毁方法 设置成我们定义的方法 测试 有创建和初始化.但是没有销毁,.对象一直没有销毁的方法 main方法是一切应用程序的入门.当main方法结束后. ...
- 如何在web.config文件中配置Session变量的生命周期
实例说明:在网上购物商城中,为了维护在线购物环境,一般只有注册会员才可以购买商品.实现购物功能时,先通过Session变量记录会员的登录名,然后在购买商品页面通过判断会员是否登录确定其能否购买商品. ...
- Flutter--Flutter中Widget、App的生命周期
前言 在App的开发过程中,我们通常都需要了解App以及各个页面的生命周期,方便我们在App进入前台时启动一些任务,在进入后台后暂停一些任务.同时,各个页面的生命周期也很重要,每个页面消失时要做一些内 ...
- C/C++——C++变量的作用域与生命周期,C语言中变量的作用域和生命周期
全局变量 作用域:全局作用域(全局变量只需在一个源文件中定义,就可以作用于所有的源文件.) 生命周期:程序运行期一直存在 引用方法:其他文件中要使用必须用extern 关键字声明要引用的全局变量. 内 ...
随机推荐
- 基本排序算法的java实现
本例子实现了一些常见的排序算法,注释中也有一些关于这些算法的思想的描述,这里不做多说,直接上代码. import java.awt.List; import java.util.ArrayList; ...
- 考虑virtual函数以外的其它选择
详情见<Effective C++>item35 1.使用non-virtual interface(NVI)手法,这是Template Method设计模式的一种特殊形式. 它以publ ...
- Java多线程(六) 线程系列总结
多线程系列终于终结得差不多,本人对该系列所做的总结大致如下: 线程锁模块耗费了大量的时间,底层的AQS实现比较复杂.仍然没有时间总结源码部分,能够坚持写下这么几个篇幅的内容真心佩服自己....希望继续 ...
- 操作xml文档的常用方式
1.操作XML文档的两种常用方式: 1)使用XmlReader类和XmlWriter类操作 XmlReader是基于数据流的,占用极少的内存,是只读方式的,所以速度极快.只能采用遍历的模式查找数据节点 ...
- 超棒的阿里巴巴矢量图标库——支持IE6
Iconfont.cn 是由阿里巴巴UX部门推出的矢量图标管理网站,也是国内首家推广Webfont形式图标的平台.网站涵盖了1000多个常用图标并还在持续更新 中,Iconfont平台为用户提供在线图 ...
- flash memory
数据删除不是以单个的字节为单位而是以固定的区块为单位(注意:NOR Flash 为字节存储.),区块大小一般为256KB到20MB. 由于其断电时仍能保存数据,闪存通常被用来保存设置信息,如在电脑的B ...
- CodeBlocks去掉拼写检查
打开: 选择Compiler... 将红框里面的勾都点掉即可!
- PHP核心代码库中的APC缓存说明123
1.APC缓存简介APC,全称是Alternative PHP Cache,官方翻译叫”可选PHP缓存”.它为我们提供了缓存和优化PHP的中间代码的框架. APC的缓存分两部分:系统缓存和用户数据缓存 ...
- 论Oracle字符集“转码”过程
本文将通过实验来演示一下Oracle字符集“转码”的确认过程. 1.实验环境说明 客户端是Windows XP操作系统的SQL*Plus程序,客户端字符集是936(对应Oracle的ZHS16GBK字 ...
- eclipse里maven项目An error occurred while filtering resources解决办法(转载)
转自:http://liyanjie918.blog.163.com/blog/static/20227290201581143110105/ 在使用eclipse构建maven项目时,突然出现错误提 ...