spring实例化bean的两种情况_prototype和singleton

当通过Spring容器创建一个Bean实例时,不仅可以完成Bean实例的实例化,还可以为Bean指定特定的作用域。Spring支持如下5种作用域,在这里主要说的就是的singleton–单例模式和prototype–原型模式,这两个Bean的作用域也是在Spring中比较常见的。

一、singleton:单例模式

在整个Spring 容器中,使用singleton定义的Bean将只有一个实例;注意这里的Spring singleton模式是默认缺省的饿汉模式,实例化spring容器时,为所有spring配置文件中定义的bean都生成一个实例。
其中核心代码如下:

class Bean {  
      private static  Bean theBean = new Bean();  
      private Bean() {}    
      public static Bean getBean() {  
          return theBean;  
      }  
 }

这样做会导致系统的开销比较大,为了不占用过大的资源或者是减少开销,一般我会xml中配置中添加缺省文件:
也叫作懒加载

<beans default-lazy-init="true">

将系统默认的饿汉模式转换成懒汉模式,也就是在第一次请求的时候我们才生成一个实例,在之后的使用中继续调用即可。
核心代码如下:

class Bean {  
      private static final Bean theBean;  
      private Bean() {}    
      public static Bean getBean() {  
          if (theBean == null) {  
              theBean = new Bean();  
          }  
          return theBean;  
      }  
 }

二、prototype:原型模式

每次通过容器的getBean方法获取prototype定义的Bean时,都将产生一个新的Bean实例。
对于singleton作用域的 Bean,每次请求该Bean都将获得相同的实例。容器负责跟踪Bean实例的状态,负责维护Bean实例的生命周期行为;如果一个Bean被设置成 prototype作用域,程序每次请求该id的Bean,Spring都会新建一个Bean实例,然后返回给程序。在这种情况下,Spring容器仅仅 使用new 关键字创建Bean实例,一旦创建成功,容器不在跟踪实例,也不会维护Bean实例的状态。
如果不指定Bean的作用域,Spring默认使用singleton作用域。Java在创建Java实例时,需要进行内存申请;销毁实例时,需要完成垃圾回收,这些工作都会导致系统开销的增加。因此,prototype作用域Bean的创建、销毁代价比较大。而singleton作用域的Bean实例一旦创建成功,可以重复使用。因此,除非必要,否则尽量避免将Bean被设置成prototype作用域。
测试代码如下:
配置Xml如下:

<!-- 默认的作用域:singleton -->
 	<bean id="orderDao" class="test.OrderDao" ></bean>
 	<!-- 指定作用域:prototype -->
 	<bean id="productDao" class="test.ProductDao" scope="prototype"></bean>

测试类如下:

ApplicationContext act=new ClassPathXmlApplicationContext(new String[] {"beans.xml"});
		//两次取OrderDao对象,OrderDao为默认的singleton作用域
		OrderDao orderDao1=(OrderDao) act.getBean("orderDao");
		OrderDao orderDao2=(OrderDao) act.getBean("orderDao");
		
		//两次取Product对象,ProductDao是prototype的作用域
		ProductDao productDao1 = (ProductDao) act.getBean("productDao");
		ProductDao productDao2 = (ProductDao) act.getBean("productDao");
		
		System.out.println("orderDao1和orderDao2是否是同一个对象 :  "+(orderDao1==orderDao2));
		System.out.println("productDao1和productDao2是否是统一个对象: "+(productDao1==productDao2));

控制台 结果 如下:
在这里插入图片描述
从结果可以看出,正如上文所述:对于singleton作用域的Bean,每次请求该id的Bean,都将返回同一个实例,而prototype作用域的Bean, 每次请求都将产生全新的实例。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

code_mo

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值