浅谈为啥 Spring 的 Bean 默认是单例的?

作者: 虹猫日志 | 来源:发表于2019-11-13 20:54 被阅读0次

前言

熟悉 Spring开发的朋友都知道现在 Spring 提供了 6 种 scope,分别是:

  • singleton
  • prototype
  • request
  • session
  • application
  • websocket

如下图是官方文档上的截图,感兴趣的朋友可以进去看看这六种分别有什么不同。今天要介绍的是这六种中的前两种,也是 Spring最初提供的 bean scope:singletonprototype

Spring 官方文档介绍如下图:

Spring bean Scope

更多内容可以看官方文档介绍,非常详细:Spring bean Scope

单例bean与原型bean的区别

如果一个 bean被声明为单例的时候,在处理多次请求的时候在 Spring 容器里只实例化出一个bean,后续的请求都共用这个对象,这个对象会保存在一个map里面。当有请求来的时候会先从 缓存(map) 里查看有没有,有的话直接使用这个对象,没有的话才实例化一个新的对象,所以这是个单例的。但是对于 原型(prototype) bean 来说当每次请求来的时候直接实例化新的bean,没有缓存以及从缓存查的过程。

1. 画图分析

singleton prototype

2. 源码分析

生成 bean时先判断单例的还是原型的

bean 判断

如果是单例的则先尝试从缓存里获取,没有再新创建

获取单例

3. 结论

  • 单例的 bean只有第一次创建新的 bean 后面都会复用该 bean,所以不会频繁创建对象。

  • 原型的 bean每次都会新创建

单例 bean 的优势

由于不会每次都新创建新对象所以有一下几个性能上的优势:

  1. 减少了新生成实例的消耗 新生成实例消耗包括两方面,第一,Spring会通过反射或者 cglib来生成 bean实例,这都是耗性能的操作,其次给对象分配内存也会涉及复杂算法。

  2. jvm垃圾回收 由于不会给每个请求都新生成bean实例,所以自然回收的对象少了。

  3. 可以快速获取到 bean 因为单例的获取bean操作除了第一次生成之外其余的都是从缓存里获取的,所以很快。

单例 bean 的劣势

单例的 bean 有一个很大的劣势就是他不能做到线程安全!!!由于所有请求都共享一个 bean实例,所以这个 bean 要是为有状态的一个 bean的话可能在\color{red}{并发场景下}出现问题,而原型的 bean则不会有这样的问题(但也有例外,比如他被单例 bean依赖),因为给每个请求都新创建实例。

总结

为了提高性能,所以 Spring 的 Bean 默认是单例的。主要为以下几点:

  1. 少创建实例
  2. 垃圾回收减少
  3. 缓存快速获取

相关文章

网友评论

    本文标题:浅谈为啥 Spring 的 Bean 默认是单例的?

    本文链接:https://www.haomeiwen.com/subject/zaykictx.html