当前位置: 首页 > 编程笔记 >

单例模式 分析代码优化方法

何晗昱
2023-03-14
本文向大家介绍单例模式 分析代码优化方法,包括了单例模式 分析代码优化方法的使用技巧和注意事项,需要的朋友参考一下

单例模式是23种设计模式之一,是比较简单的一种设计模式,它的目的是无论调用多少次,都返回同一个对象,它的特点是构造器私有化。

  它分为两种结构,一种是懒汉式的,一种是饿汉式的,它们各有优缺点,我们先从饿汉式看起,代码如下:

public class Single { 
  private static Single single = new Single();  
  private Single() {  
  } 
   public Single getInstance() { 
    return single; 
  }  
}

通过上面的程序可以看出来虽然我们加载同一个对象的目的确实达到了,但当程序被加载的时候就会创建single这个对象,当这个类有多个这样的方法时,我们可能会用不到这个对象中大多数单例,就会造成对内存的浪费。所以就出现了懒汉式的单例模式,代码如下:

public class Single { 
  private static Single single = null;  
  private Single() {  
  } 
 
  public Single getInstance() { 
    if(single==null){ 
      single = new Single(); 
    } 
    return single; 
  }  
}

这样,只有当我们真正调用这个对象时它才会被new出来,但是这样是存在问题的。

  当上面的第二段代码在第一次加载的时候有两个线程对其进行了调用,则会产生两个不同的对象,所以是线程不安全的,这时候就会想到给这个方法加个锁,加锁之后的代码如下:

public class Single { 
  private static Single single = null;  
  private Single() {  
  } 
 
  public synchronized Single getInstance() { 
    if (single == null) { 
      single = new Single(); 
    } 
    return single; 
  }  
}

这样做确实做到了线程安全,但是当加锁这个方法里面要执行很多东西,调用这个方法花费的时间会很长,这样对服务器来说是致命的,因为这个方法如果某个线程一直调用的话,其它的线程是没有办法调的,服务器就阻塞了,那么升级后的代码如下:

public class Single { 
  priate static Single single = null;  
  private Single() {  
  } 
 
  public Single getInstance() { 
    if (single == null) { 
      synchronized (Single.class) { 
        single = new Single(); 
      } 
    } 
    return single; 
  } 
}

仔细观察以后发现这样并没有锁住,当第一次同时有两个线程到达getInstance()方法if判断时,其中有一个肯定是阻塞的,当另外一个执行完以后,阻塞这个线程是不会再判断是否为空的,还是会创建一个对象的,这样又有多个对象被产生了,再对其进行升级,得到的代码如下:

public class Single { 
  private static Single single = null;  
  private Single() {  
  } 
 
  public Single getInstance() { 
    if (single == null) { 
      synchronized (Single.class) { 
        if (single == null) { 
          single = new Single(); 
        } 
      } 
    } 
    return single; 
  } 
}

这样就不会产生上面的问题,而且也只锁一次,因为第二次再执行这个方法时,会跳过if判断,直接返回single,不会再被锁,执行效率也会很高。

  但即使是这样,也还是有问题的,因为我们不能确定在内存中是先给对象赋值,还是先创建了这个对象,所以第二个程序有可能得到的是初始化一半了的对象,在jdk1.5之后,我们可以用volatile这个关键字来避免这种情况,代码如下:

public class Single { 
  private static volatile Single single = null;  
  private Single() {  
  } 
 
  public Single getInstance() { 
    if (single == null) { 
      synchronized (Single.class) { 
        if (single == null) { 
          single = new Single(); 
        } 
      } 
    } 
    return single; 
  } 
}

但是这种情况很少使用,我在这里只是为了学习一下,嘻嘻

 类似资料:
  • 本文向大家介绍php单例模式实现方法分析,包括了php单例模式实现方法分析的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了php单例模式实现方法。分享给大家供大家参考。具体如下: 希望本文所述对大家的php程序设计有所帮助。

  • 本文向大家介绍Python单例模式实例分析,包括了Python单例模式实例分析的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了Python单例模式的使用方法。分享给大家供大家参考。具体如下: 方法一 1.禁用__init__方法,不能直接创建对象。 2.__instance,单例对象私有化。 3.@staticmethod,静态方法,通过类名直接调用。 4.__lock,代码锁。 5.继承

  • 本文向大家介绍Java单例模式、饥饿模式代码实例,包括了Java单例模式、饥饿模式代码实例的使用技巧和注意事项,需要的朋友参考一下

  • 本文向大家介绍php设计模式之单例模式代码,包括了php设计模式之单例模式代码的使用技巧和注意事项,需要的朋友参考一下 php设计模式之单例模式的例子,供大家参考,具体内容如下 以上就是本文的全部内容,希望对大家学习php程序设计有所帮助。

  • 本文向大家介绍通过示例分析Swift单例模式,包括了通过示例分析Swift单例模式的使用技巧和注意事项,需要的朋友参考一下 三种Swift实现单例模式的方法:全局变量,内部变量,dispatch_once方式 1. 全局变量 2. 内部变量 3. dispatch_once方式 以上所述就是本文的全部内容了,希望大家能够喜欢。

  • 本文向大家介绍JAVA 枚举单例模式及源码分析的实例详解,包括了JAVA 枚举单例模式及源码分析的实例详解的使用技巧和注意事项,需要的朋友参考一下 JAVA 枚举单例模式及源码分析的实例详解       单例模式的实现有很多种,网上也分析了如今实现单利模式最好用枚举,好处不外乎三点: 1.线程安全 2.不会因为序列化而产生新实例 3.防止反射攻击但是貌似没有一篇文章解释ENUM单例如何实现了上述三