延迟初始化
一般有几种延迟初始化的场景:
- 对于会消耗较多资源的对象:这不仅能够节省一些资源,同时也能够加快对象的创建速度,从而从整体上提升性能。
- 某些数据在启动时无法获取:比如一些上下文信息可能在其他拦截器或处理中才能被设置,导致当前bean在加载的时候可能获取不到对应的变量的值,使用 延迟初始化可以在真正调用的时候去获取,通过延迟来保证数据的有效性。
在Java8中引入的lambda对于我们实现延迟操作提供很大的便捷性,如Stream、Supplier等,下面介绍几个例子。
Lambda
Supplier
通过调用get()方法来实现具体对象的计算和生成并返回,而不是在定义Supplier的时候计算,从而达到了_延迟初始化_的目的。但是在使用 中往往需要考虑并发的问题,即防止多次被实例化,就像Spring的@Lazy注解一样。
- public class Holder {
- // 默认第一次调用heavy.get()时触发的同步方法
- private Supplier<Heavy> heavy = () –> createAndCacheHeavy();
- public Holder() {
- System.out.println("Holder created");
- }
- public Heavy getHeavy() {
- // 第一次调用后heavy已经指向了新的instance,所以后续不再执行synchronized
- return heavy.get();
- }
- //…
- private synchronized Heavy createAndCacheHeavy() {
- // 方法内定义class,注意和类内的嵌套class在加载时的区别
- class HeavyFactory implements Supplier<Heavy> {
- // 饥渴初始化
- private final Heavy heavyInstance = new Heavy();
- public Heavy get() {
- // 每次返回固定的值
- return heavyInstance;
- }
- }
- //第一次调用方法来会将heavy重定向到新的Supplier实例
- if(!HeavyFactory.class.isInstance(heavy)) {
- heavy = new HeavyFactory();
- }
- return heavy.get();
- }
- }
当Holder的实例被创建时,其中的Heavy实例还没有被创建。下面我们假设有三个线程会调用getHeavy方法,其中前两个线程会同时调用,而第三个线程会在稍晚的时候调用。
当前两个线程调用该方法的时候,都会调用到createAndCacheHeavy方法,由于这个方法是同步的。因此第一个线程进入方法体,第二个线程开始等待。在方法体中会首先判断当前的heavy是否是HeavyInstance的一个实例。
如果不是,就会将heavy对象替换成HeavyFactory类型的实例。显然,第一个线程执行判断的时候,heavy对象还只是一个Supplier的实例,所以heavy会被替换成为HeavyFactory的实例,此时heavy实例会被真正的实例化。
等到第二个线程进入执行该方法时,heavy已经是HeavyFactory的一个实例了,所以会立即返回(即heavyInstance)。当第三个线程执行getHeavy方法时,由于此时的heavy对象已经是HeavyFactory的实例了,因此它会直接返回需要的实例(即heavyInstance),和同步方法createAndCacheHeavy没有任何关系了。