`

设计模式之单例模式

 
阅读更多

单例模式

核心作用
  • 保证一个类中只有一个实例,并且提供一个访问该实例的全局访问点 
  • 项目中的配置文件,一般也只有一个对象,没必要每次都都new一遍,配置文件不容易改变
  • Spring中bean的加载,以及控制器对象也是单例模式 

单例模式五种实现方式

  • 饿汗式:编程安全,调用效率高。但是不能延时加载
  • 懒汉式:线程安全,调用效率不高,但是可以延时加载
  • 双重检测锁式(由于JVM底层内部模型原因,偶尔出现问题,不建议使用)
  • 静态内部类式:线程安全,调用效率高,但是可以延时加载
  • 枚举单例:线程安全,调用效率高,不能延时加载

代码

 

 饿汗式 :线程安全的,类初始化时立即加载,天然线程安全

 

public class SingleEHan {
    private static SingleEHan instance = new SingleEHan(); //类实例化,立即加载                                          //,天然线程安全
    private SingleEHan(){} //单例模式构造器方法必须私有
    public static SingleEHan getInstance(){  //不需要同步,显然调用的效率高
        return instance;
    }
}

Compiled from "SingleEHan.java"
public class com.xhd.admin.web.SingleEHan {
  public static com.xhd.admin.web.SingleEHan getInstance();
    Code:
       0: getstatic     #2  // Field instance:Lcom/xhd/admin/web/SingleEHan;
       3: areturn

  static {};
    Code:
       0: new           #3  // class com/xhd/admin/web/SingleEHan
       3: dup
       4: invokespecial #4  // Method "<init>":()V
       7: putstatic     #2  // Field instance:Lcom/xhd/admin/web/SingleEHan;
      10: return
}
 

 

 懒汉式加载 资源利用效率高,但调用效率低

 

public class SingleLanHan {

    private static SingleLanHan singleLanHan;
    private SingleLanHan(){
        if(singleLanHan != null){    //防止反射
            throw new RuntimeException();
        }
    }

   // synchronized 调用效率低
    public static synchronized SingleLanHan getSingleLanHan(){  
        if(singleLanHan == null){
            singleLanHan = new SingleLanHan();
        }
        return singleLanHan;
    }
}

 

  静态内部类实现方式:,线程安全调用效率高 懒加载

 

public class SingleStatic {
    private SingleStatic() {}
    //静态内部类
    private static class SingleStaticInstance{
        private static final SingleStatic instance 
                                 = new SingleStatic();
    }
    public static SingleStatic getInstance(){
        return SingleStaticInstance.instance;
    }
}


Compiled from "SingleStatic.java"
public class com.xhd.admin.web.SingleStatic {
  public static com.xhd.admin.web.SingleStatic getInstance();
    Code:
       0: invokestatic  #3                  
// Method com/xhd/admin/web/SingleStatic$SingleStaticInstance.access$100:()Lcom/xhd/admin/web/SingleStatic;
       3: areturn

  com.xhd.admin.web.SingleStatic(com.xhd.admin.web.SingleStatic$1);
    Code:
       0: aload_0
       1: invokespecial #1                  // Method "<init>":()V
       4: return
}

 

枚举:天然线程安全,天然实例唯一
public enum SingleEnum {

    //这个枚举元素,本身就是单例对象
    INSTANCE;
    //添加自己需要的操作
    public void singletonOperation(){

    }
}
双重检查锁实现单例模式
public class SingleDoubleCheck {

    private static SingleDoubleCheck singleDoubleCheck = null;
    private SingleDoubleCheck(){}

    public static SingleDoubleCheck getSingleDoubleCheck(){
        if(singleDoubleCheck == null){
            SingleDoubleCheck sc ;
            synchronized (SingleDoubleCheck.class){
                sc = singleDoubleCheck;
                if(sc == null){
                    sc = new SingleDoubleCheck();
                }
            }
            singleDoubleCheck = sc;
        }
        return singleDoubleCheck;
    }
}
多线程环境下测试用CountDownLatch
  • 同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待
  • countDown() 当前线程调此方法,则统计减一,建议放在finally里执行
  • await方法:调用此方法会一直阻塞当前线程,直到计数器的值为0 
public class ClientTest {

    public static void main(String[] args) throws InterruptedException {
        long start = System.currentTimeMillis();
        int threadNum = 10;
        final CountDownLatch countDownLatch = new CountDownLatch(threadNum);
        for(int j = 0; j< threadNum;j++){
            new Thread(new Runnable() {
                @Override
                public void run() {
                    for(int i = 0;i<1000000;i++){
                        Object o = SingleEHan.getInstance(); //换方式在此地方
                        countDownLatch.countDown();
                    }
                }
            }).start();
        }

        countDownLatch.await();  //await 自动检测
        long end = System.currentTimeMillis();
        long between = end - start;
        System.out.println("总耗时:"+between);
    }
}

 

序列化与反序列化问题,以及反射问题

public class SingleLanHanSerial implements Serializable{
    private static SingleLanHanSerial singleLanHan;
    private SingleLanHanSerial(){
        if(singleLanHan != null){    //防止反射
            throw new RuntimeException();
        }
    }
    public static synchronized SingleLanHanSerial getSingleLanHan(){  
                                           // synchronized 调用效率低
        if(singleLanHan == null){
            singleLanHan = new SingleLanHanSerial();
        }
        return singleLanHan;
    }

// 反序列化,如果定义了readResolve方法,就直接返回Instance实例,防止了反序列化的漏洞
    private Object readResolve() throws ObjectStreamException {
        return singleLanHan;
    }
}

 

反射测试序列化测试

public class ClientEnum {
    public static void main(String[] args) throws Exception {
        //通过反射机制创造对象
        Class<SingleLanHan> lanHanClazz = (Class<SingleLanHan>)Class.forName("com.jenny.mole.single.SingleLanHan");
        Constructor<SingleLanHan> constructor = lanHanClazz.getDeclaredConstructor(null);
        constructor.setAccessible(true); //可以访问私有方法了这样
        SingleLanHan singleLanHan = constructor.newInstance();
        SingleLanHan singleLanHan1 = constructor.newInstance();
        System.out.println(singleLanHan);
        System.out.println(singleLanHan1);

//        通过反序列化的方式创造对象
        SingleLanHanSerial single2 = SingleLanHanSerial.getSingleLanHan();
        FileOutputStream fos = new FileOutputStream("/d:/a.txt");
        ObjectOutputStream oos = new ObjectOutputStream(fos);
        oos.writeObject(single2);

        ObjectInputStream ois = new
                       ObjectInputStream(new FileInputStream("/d:/a.txt"));
        SingleLanHanSerial s3 = (SingleLanHanSerial)ois.readObject();

        System.out.println(single2);
        System.out.println(s3);

    }
}

 

 

静态内部类

外部类没有static属性,则不会像懒汉式那样立即加载对象只有真正调用getInstance,才会加载静态内部类,加载类时时线程安全的,Instance是static final类型,保证了内存中只有一个实例存在,而且只能被赋值一次,从而保证了线程安全性兼备了并发高效调用和延迟加载的优势

 

枚举方式

  • 优点:实现简单,枚举本身就是单例模式,由JVM从根本上提供保障,避免通过反射和反序列化的漏洞
  • 缺点:无延迟加载

如何选用
  • 单例对象,占用资源少,不需要延时加载:枚举好于 懒汉式
  • 单例对象占用资源大,需要延时加载:静态内部类好于懒汉式 
分享到:
评论

相关推荐

    Java设计模式之单例模式的七种写法

    Java设计模式之单例模式的七种写法 单例模式是一种常见的设计模式,它确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例。在计算机系统中,线程池、缓存、日志对象、对话框、打印机的驱动程序对象常...

    python 设计模式之单例模式

    python 设计模式之单例模式

    设计模式之单例模式详解.pdf

    设计模式之单例模式详解 单例模式是一种常用的软件设计模式,其定义是单例对象的类只能允许一个实例存在。许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为。 单例模式的实现主要是...

    研磨设计模式之单例模式

    通过研磨设计模式之单例模式的资料,你可以深入理解单例模式的原理、实现方式及其优缺点,进一步提升自己的编程技能和设计思维。学习并熟练掌握设计模式,对于成为一名优秀的Java开发者至关重要。

    java设计模式之单例模式详解

    Java设计模式之单例模式详解 一、单例模式概览 单例模式(Singleton Pattern)是面向对象设计模式中的一种,属于创建型模式。它确保一个类仅有一个实例,并提供一个全局访问点来访问该实例。单例模式的核心在于控制...

    设计模式之单例模式(结合工厂模式)

    单例模式是软件设计模式中的一种经典模式,它保证了类只有一个实例存在,并提供一个全局访问点。在Java等面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。结合工厂模式,...

    php设计模式之单例模式_.docx

    ### PHP设计模式之单例模式详解 #### 一、引言 在软件工程领域,设计模式是一种被广泛接受的解决方案,用于解决特定类型的问题。PHP作为一种流行的服务器端脚本语言,同样可以从这些设计模式中受益。本文将详细介绍...

    JAVA设计模式之单例模式(完整版)1[定义].pdf

    JAVA设计模式之单例模式(完整版)1[定义].pdf

    23钟设计模式之单例模式

    单例模式是一种常用的设计模式,它的核心思想是在整个应用程序中,一个类只能有一个实例存在。单例模式常用于控制资源的共享,例如数据库连接池、日志服务等。单例模式有多种实现方式,常见的包括懒汉式、饿汉式以及...

    设计模式之单例模式和工厂模式

    细心整合和单例模式和工厂模式的几种模型,懒汉式,饿汉式,如何并发操作模式,等都有详细讲解

    设计模式之单例模式.md

    设计模式之单例模式,单列模式的几种实现形式,以及其优缺点,还有就是示例,对初步了解单列模式的有所帮助

    设计模式之单例模式.pptx

    单例模式是一种常用的软件设计模式,其定义是单例对象的类只能允许一个实例存在。 许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为。 比如在某个服务器程序中,该服务器的配置信息存放...

    设计模式之单例模式Java实现和类设计图

    本次我们将深入探讨两种设计模式——单例模式和装饰模式,它们在Java编程中都有着广泛的应用。 首先,让我们来理解“单例模式”。单例模式是一种创建型设计模式,其核心思想是保证一个类只有一个实例,并提供一个...

    Java设计模式之单例模式

    目录 单例模式的概念 单例模式的要点 单例模式类图 单例模式归类 单例模式的应用场景 单例模式解决的问题 单例模式的实现方式 单例模式实现方式对比 单例模式的概念 单例模式,顾名思义就是只有一个实例,并且由它...

Global site tag (gtag.js) - Google Analytics