有一天我拉屎的时候想的一个问题,Spring为啥默认把bean设计成单例的?

Bean简介:

       在Spring中,那些组成你应用程序的主体(backbone)及由Spring IoC容器所管理的对象,被称之为bean。 简单地讲,bean就是由Spring容器初始化、装配及管理的对象,除此之外,bean就与应用程序中的其他对象没有什么区别了。 而bean定义以及bean相互间的依赖关系将通过配置元数据来描述。

Bean的作用域

       创建一个bean定义,其实质是用该bean定义对应的类来创建真正实例的“配方(recipe)”。把bean定义看成一个配方很有意义,它与class很类似,只根据一张“处方”就可以创建多个实例。

       你不仅可以控制注入到对象中的各种依赖和配置值,还可以控制该对象的作用域。这样你可以灵活选择所建对象的作用域,而不必在Java Class级定义作用域。Spring Framework支持五种作用域(其中有三种只能用在基于web的Spring ApplicationContext)。

1、singleton作用域:

       Spring的scope的默认值是singleton

       Spring 只会为每一个bean创建一个实例,并保持bean的引用.

 或 

 

2、prototype作用域:

       一个bean定义对应多个对象实例。

       每一次请求(将其注入到另一个bean中,或执行getBean()方法)都会产生一个新的bean实例,相当于new操作.

       Spring一旦将Bean实例交给(注入)调用者,就不再持有这个bean的引用。就无法再执行bean定义的destroy-method.

       清除prototype作用域的bean对象并释放资源,是调用者的职责。

 或 

3、request作用域:

       HTTP request表示该针对每一次HTTP请求都会产生一个新的bean,仅适用于WebApplicationContext环境。

4、session作用域:

       HTTP session表示该针对每一次HTTP请求都会产生一个新的bean,仅适用于WebApplicationContext环境。

5、globalSession作用域:

       在一个全局的HTTP Session中,一个bean定义对应一个实例。典型情况下,仅在使用portlet context的时候有效。该作用域仅在基于web的Spring ApplicationContext情形下有效。

       考虑下面bean定义:

6、自定义scope,必须实现ScopeMetadataResolve接口提供的无参数构造器


        

7、scope举例

       可以通过hashCode判读是否为同一个Bean

@Scope
@Component
public class BeanAnnotation {
    
    public void say(String arg) {
        System.out.println("BeanAnnotation : " + arg);
    }
    
    public void myHashCode() {
        System.out.println("BeanAnnotation : " + this.hashCode());
    }
    
}

今天我们来延伸一下单例bean与原型bean的区别:

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

1.画图分析

有一天我拉屎的时候想的一个问题,Spring为啥默认把bean设计成单例的?_第1张图片

有一天我拉屎的时候想的一个问题,Spring为啥默认把bean设计成单例的?_第2张图片

2.源码分析

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

有一天我拉屎的时候想的一个问题,Spring为啥默认把bean设计成单例的?_第3张图片

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

有一天我拉屎的时候想的一个问题,Spring为啥默认把bean设计成单例的?_第4张图片

结论:

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

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

单例bean的优势

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

1.减少了新生成实例的消耗

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

2.减少jvm垃圾回收

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

3.可以快速获取到bean

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

单例bean的劣势

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

总结

Spring 为啥把bean默认设计成单例?

答案:为了提高性能!!!从几个方面,

  1. 少创建实例

  2. 垃圾回收

  3. 缓存快速获取

单例有啥劣势?

如果是有状态的话在并发环境下线程不安全

你可能感兴趣的:(Spring)