博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
spring中的设计模式(四)
阅读量:6121 次
发布时间:2019-06-21

本文共 2008 字,大约阅读时间需要 6 分钟。

5.单例模式

首先单例模式中的懒汉和饿汉模式,

懒汉模式:

//懒汉式单例类.在第一次调用的时候实例化自己

public class Singleton {

   private Singleton() {}

   private static Singleton single=null;

   //静态工厂方法

   public static Singleton getInstance() {

        if (single == null) { 

            single = new Singleton();

        } 

       return single;

    }

}

但是这个懒汉模式并没有考虑到线程安全,所以可以有三种方法对这个懒汉模式进行线程安全化

1.给getInstance方法加synchronized 关键字

2.双重检查锁定

public static Singleton getInstance() {

       if (singleton == null) { 

           synchronized (Singleton.class) { 

               if (singleton == null) { 

                  singleton = new Singleton();

               } 

           } 

       } 

       return singleton;

    }

3.静态内部类的方式

public class Singleton { 

   private static class LazyHolder { 

      private static final Singleton INSTANCE = new Singleton(); 

   } 

   private Singleton (){} 

   public static final Singleton getInstance() { 

      return LazyHolder.INSTANCE; 

   } 

饿汉模式:

//饿汉式单例类.在类初始化时,已经自行实例化

public class Singleton1 {

   private Singleton1() {}

   private static final Singleton1 single = new Singleton1();

   //静态工厂方法

   public static Singleton1 getInstance() {

       return single;

    }

}

 

区别:

饿汉模式天生就是线程安全的,而懒汉模式则需要自行实现线程安全

饿汉式模式在类创建的时候就会实例化一个静态的对象出来,会占用一定内存,但是相应的第一次调用会快一点

懒汉模式的延迟加载,需要调用的时候才会实例化对象

第一种方法实现同步的话,每次调用都会同步,非常耗性能,因为大部分情况是不需要同步的

第二种方法双重判断则避免了每次都需要同步的情况

第三种方法,利用了ClassLoader的机制确保初始化Intance的时候都只有一个线程

 

spring中的单例模式:

spring中的依赖注入都发生在AbstractBeanFactory的getBean方法里面,而getBean方法里面doGetBean方法中的getSingleton方法

 

protected Object getSingleton(String beanName, boolean allowEarlyReference)

    {

        Object singletonObject = singletonObjects.get(beanName);

        if(singletonObject == null &&isSingletonCurrentlyInCreation(beanName))

            synchronized(singletonObjects)

            {

                singletonObject = earlySingletonObjects.get(beanName);

                if(singletonObject == null && allowEarlyReference)

                {

                    ObjectFactory singletonFactory =(ObjectFactory)singletonFactories.get(beanName);

                    if(singletonFactory != null)

                    {

                        singletonObject = singletonFactory.getObject();

                        earlySingletonObjects.put(beanName, singletonObject);

                        singletonFactories.remove(beanName);

                    }

                }

            }

        return singletonObject == NULL_OBJECT ? null : singletonObject;

   }

 

可以看到spring中用的是懒汉模式的双重判断来实现线程安全的,避免在加锁的瞬间有其他注入的时候创建实例

转载地址:http://vrwua.baihongyu.com/

你可能感兴趣的文章
我的友情链接
查看>>
PCS子层有什么用?
查看>>
查看端口,关闭端口
查看>>
代码托管平台简介
查看>>
linux:yum和apt-get的区别
查看>>
Sentinel 1.5.0 正式发布,引入 Reactive 支持
查看>>
如何对网站进行归档
查看>>
数据库之MySQL
查看>>
2019/1/15 批量删除数据库相关数据
查看>>
数据类型的一些方法
查看>>
Mindjet MindManager 2019使用教程:
查看>>
游戏设计的基本构成要素有哪些?
查看>>
详解 CSS 绝对定位
查看>>
AOP
查看>>
我的友情链接
查看>>
NGUI Label Color Code
查看>>
.NET Core微服务之基于Polly+AspectCore实现熔断与降级机制
查看>>
vue组件开发练习--焦点图切换
查看>>
浅谈OSI七层模型
查看>>
Webpack 2 中一些常见的优化措施
查看>>