【Spring进阶指南】Spring 为啥默认把bean设计成单例的?

上次分享了Redis相关问题的分享, 【Redis面试题】Redis的字符串是怎么实现的?以后今天想在分享一篇有关spring bean相关的内容。原问题大概是先问了我spring bean做用域相关问题,最后问了我spring 为啥默认把bean设计成单例的?
我打算从如下几方面讲起:html

  • 单例bean与原型bean的区别
  • 单例bean的优点
  • 单例bean的劣势
  • 总结

熟悉spring开发的朋友都知道spring 提供了5种scope分别是singleton, prototype, request, session,global session。以下图是官方文档上的截图,感兴趣的朋友能够进去看看这五种分别有什么不一样。今天要介绍的是这五种中的前两种,也是spring最初提供的bean scope singleton 和 prototype。java

spring 官方文档介绍以下图: 面试

更多内容能够看官方文档介绍,很是详细: Spring bean做用域

单例bean与原型bean的区别

若是一个bean被声明为单例的时候,在处理屡次请求的时候在spring 容器里只实例化出一个bean,后续的请求都公用这个对象,这个对象会保存在一个map里面。当有请求来的时候会先从缓存(map)里查看有没有,有的话直接使用这个对象,没有的话才实例化一个新的对象,因此这是个单例的。可是对于原型(prototype)bean来讲当每次请求来的时候直接实例化新的bean,没有缓存以及从缓存查的过程。算法

1.画图分析spring

2.源码分析缓存

生成bean时先判断单例的仍是原型的安全

若是是单例的则先尝试从缓存里获取,没有在新建立session

结论:

1.单例的bean只有第一次建立新的bean 后面都会复用该bean,因此不会频繁建立对象。并发

2.原型的bean每次都会新建立jvm

单例bean的优点

因为不会每次都新建立新对象因此有一下几个性能上的优点: 1.减小了新生成实例的消耗 新生成实例消耗包括两方面,第一,spring会经过反射或者cglib来生成bean实例这都是耗性能的操做,其次给对象分配内存也会涉及复杂算法

2.减小jvm垃圾回收 因为不会给每一个请求都新生成bean实例,因此天然回收的对象少了

3.能够快速获取到bean 由于单例的获取bean操做除了第一次生成以外其他的都是从缓存里获取的因此很快

有关bean实例化相关能够看着篇文章:Spring 源码分析之 bean 实例化原理

单例bean的劣势

单例的bean一个很大的劣势就是他不能作到线程安全!!!,因为全部请求都共享一个bean实例,因此这个bean要是有状态的一个bean的话可能在并发场景下出现问题,而原型的bean则不会有这样问题(但也有例外,好比他被单例bean依赖),由于给每一个请求都新建立实例。关于这方面我正在准备写一篇文章,在整理当中,感兴趣的朋友能够关注我,我后续写一篇详细的文章。

总结

Spring 为啥把bean默认设计成单例?
答案:为了提升性能!!!从几个方面,1.少建立实例2.垃圾回收3.缓存快速获取
单例有啥劣势?
若是是有状态的话在并发环境下线程不安全

有关spring bean的线程安全相关话题,我会在下一篇文章中写出。



欢迎加群一块儿讨论MySQL,Spring , java等。


其余文章

【Redis面试题】Redis的字符串是怎么实现的?

Spring 源码分析之 bean 实例化原理?

Spring 源码分析之 bean 依赖注入原理(注入属性)

Spring源码分析之 lazy-init 实现原理

相关文章
相关标签/搜索