前言

上面的章节中,我们介绍了工厂模式,它是创建型模式的一种。本章我们将会介绍 单例模式 ,它也是创建型模式的一种。单例模式是我们比较常用的一个设计模式,也是最简单的一种设计模式。

单例模式

介绍:确保在一个jvm中某个类有且只有一个实例。

类图:

从上面的类图中,我们可以总结出以下几点:

  • 单例模式需要实现一个私有的构造函数,防止有外部的语句会实例化该类。
  • 单例模式需要定义一个静态成员变量,用来指向该类的唯一实例。
  • 单例模式需要定义一个静态成员方法,用来返回该类唯一的实例。

单例模式的实现

单例模式的实现方式有8种,下面让我们分别来看一看这几种实现方式的区别。

饿汉模式

代码:

public class Singleton {
private Singleton() {
    if (singleton != null) {
      throw new RuntimeException();
    }
  }
private static Singleton singleton = new Singleton();
public static Singleton initSingleton() {
return singleton;
}
}
  • 是否是懒加载:否
  • 是否线程安全:是
  • 优点:不用加锁,执行效率高。
  • 缺点:类加载时就初始化,浪费内存。

饿汉模式(静态代码块初始化)

代码:

public class Singleton {
private Singleton() {
    if (singleton != null) {
      throw new RuntimeException();
    }
  }
private static Singleton singleton;
static {
singleton = new Singleton();
}
public static Singleton initSingleton() {
return singleton;
}
}
  • 是否是懒加载:否
  • 是否线程安全:是
  • 优点:不用加锁,执行效率高。
  • 缺点:类加载时就初始化,浪费内存。

懒汉模式

import java.util.Objects;

public class Singleton {
private Singleton() {
    if (singleton != null) {
      throw new RuntimeException();
    }
  }
private static Singleton singleton;
public static Singleton initSingleton() {
if (Objects.isNull(singleton)) {
singleton = new Singleton();
}
return singleton;
}
}
  • 是否是懒加载:是
  • 是否线程安全:否
  • 优点:不用加锁,执行效率高,第一次调用时初始化,避免了浪费内存。
  • 缺点:非线程安全,多线程环境下,可能会产生多个实例。

懒汉模式(线程安全)

import java.util.Objects;

public class Singleton {
private Singleton() {
    synchronized(Singleton.class) {
      if (Objects.nonNull(singleton)) {
        throw new RuntimeException();
      }
    }
  }
private static Singleton singleton;
public static synchronized Singleton initSingleton() {
if (Objects.isNull(singleton)) {
singleton = new Singleton();
}
return singleton;
}
}
  • 是否是懒加载:是
  • 是否线程安全:是
  • 优点:第一次调用时初始化,避免了浪费内存。
  • 缺点:实现了synchronized方法,会影响执行效率。

懒汉模式(同步代码块)

import java.util.Objects;

public class Singleton {
private Singleton() {
    synchronized(Singleton.class) {
      if (Objects.nonNull(singleton)) {
        throw new RuntimeException();
      }
    }
  }
private static Singleton singleton;
public static Singleton initSingleton() {
if (Objects.isNull(singleton)) {
synchronized (Singleton.class) {
singleton = new Singleton();
}
}
return singleton;
}
}
  • 是否是懒加载:是
  • 是否线程安全:否
  • 优点:第一次调用时初始化,避免了浪费内存。
  • 缺点:虽然实现了synchronized代码块,但是如果有多个线程同时实例化,那么有可能会产生多个实例。

双重检查模式

import java.util.Objects;

// 不能加Serializable,防止反序列化创建对象
public class Singleton {
private Singleton() {
    // 防止反射创建对象
    if (Objects.isNull(singleton)) {
      synchronized (Singleton.class) {
        if (Objects.isNull(singleton)) {
          singleton = this;
        } else {
          throw new RuntimeException();
        }
      }
    } else {
      throw new RuntimeException();
    }
  }
private static volatile Singleton singleton;
public static Singleton initSingleton() {
if (Objects.isNull(singleton)) {
synchronized (Singleton.class) {
if(Objects.isNull(singleton)) {
singleton = new Singleton();
}
}
}
return singleton;
}
}
  • 是否是懒加载:是
  • 是否线程安全:是
  • 优点:第一次调用时初始化,避免了浪费内存。
  • 缺点:暂无

静态内部类

public class Singleton {
private Singleton() {
    if (Objects.nonNull(singleton)) {
      throw new RuntimeException();
    }
  }
public static Singleton initSingleton() {
return SingletonInstance.singleton;
} public static class SingletonInstance {
private static Singleton singleton = new Singleton();
}
}
  • 是否是懒加载:是
  • 是否线程安全:是
  • 优点:第一次调用时初始化,避免了浪费内存,效率高。
  • 缺点:暂无

枚举

public enum Singleton {
INSTANCE;
}
  • 是否是懒加载:是
  • 是否线程安全:是
  • 优点:第一次调用时初始化,避免了浪费内存,效率高。
  • 缺点:暂无

最新文章

  1. Android带加减的edittext
  2. ROS机器人语音交互(一)
  3. 关于js预编译以及js文件执行顺序的几个问题。
  4. 手机站点动态效果插件TouchSlide
  5. Thinkpad 笔记本VMware Workstation 安装虚拟机出现“此主机支持 Intel VT-x,但 Intel VT-x 处于禁用状态”解决方法
  6. linux把EDT时间修改为CST格式
  7. UNC path
  8. centos6.2下安装redis和phpredis扩展,亲测好用
  9. 提高iOS开发效率的方法和工具
  10. structs2使用小结2
  11. 《Android底层接口与驱动开发技术详解》digest
  12. 完全用Linux工作,抛弃windows
  13. DOM编程从入门到忘记
  14. 在MySQL中使用子查询
  15. 反射:修改请求头HttpWebRequest/Webclient Header属性的date值-"此标头必须使用适当的属性进行修改"
  16. 编译安装LAMP
  17. Basic remains POJ - 2305 同余模 高精度处理
  18. 客户端如何访问访问oracle 12c 64位的数据库
  19. 精彩看点 | GIAC大会PPT+视频合集全量放送!
  20. Centos7搭建软路由

热门文章

  1. Windown Server 2008配置tomcat9虚拟路径
  2. jQuery-点击按钮页面滚动到顶部,底部,指定位置
  3. MySQL中的索引优化
  4. Swoole开启守护进程后如何关闭
  5. springboot热部署设置
  6. springboot在集成mybatis的时候老是报错 The server time zone value '�й���׼ʱ��' is unrecognized
  7. 第四小节之Java 集合类
  8. maven参数详解
  9. 【BZOJ1999】树网的核
  10. event loop 与 vue