为什么Spring中的bean默认都是单例模式?
省流小助手:主要是为了提升性能。希望知道原理的小伙伴继续往下看。
大家都知道Spring中的bean默认情况下都是单例模式(Singleton),那为什么呢?想要弄明白这个问题,首先需要对反射有一定的了解。
在Spring中,bean基本都是通过反射的方式创建的。反射可以动态的创建和使用对象,降低系统耦合,提高代码的复用率。因此反射十分适合于Spring这种注重自身拓展性的框架。但是,反射有一个致命的缺点,效率太低了。
那么接下来,让我们通过一个简单的实验来看看反射和通过new创建对象的效率对比:
long start = System.currentTimeMillis();
for (int i = 0; i < 1000000000; i++) {
Core core = new Core();
}
long end = System.currentTimeMillis();
System.out.println("new耗时:" + (end - start));
start = System.currentTimeMillis();
Class<Core> clazz = Core.class; //获取全限定类名
Constructor<Core> declaredConstructor = clazz.getDeclaredConstructor(); //获取构造函数
for (int i = 0; i < 1000000000; i++) {
declaredConstructor.newInstance();
}
end = System.currentTimeMillis();
System.out.println("反射耗时:" + (end - start));
上面分别统计了通过new和反射创建1000000000个Core(我自己定义的)对象消耗的时间,结果如下。由此可知,反射创建对象的效率是很低的。这也正是Spring为什么要用单例模式的原因,主要是为了减少对象创建来提升性能。
Spring对象默认单例的优点
- 通过减少对象创建的次数,降低反射的低效率对于框架整体性能的影响。
- 对象创建的少,GC自然也就少了,提高性能。
- 可以更快速的获取到bean。这个涉及到Spring的三级缓存,只有在第一次创建对象时才会去创建,其余的都是从第一级缓存直接获取,提高性能。
Spring对象默认单例的缺点
- 会带来线程安全问题,有状态bean(能存储数据的bean)在单例模式下线程不安全。因此,Spring也提供了其他多种bean的作用域。