《重学Java设计模式》之 单例模式

单例模式主要解决的是,一个全局使用的类频繁的创建和消费,从而提升提升整体的代码的性能。

单例模式原则

  • 私有构造。(阻止类被常规方法实例化)
  • 以静态方法或者枚举返回实例。(保证实例的唯一性)
  • 确保实例只有一个,尤其是多线程环境。
  • 确保反序列化时不会重新构建对象。

懒汉模式(线程安全)

在初次调用时构建实例 (双重检查)。

锁不加在方法上,因为如果加在方法上,多线程访问时只有一个线程能执行该方法。

在synchronized 代码块中再次检查实例是否被创建,可能同时有两个线程A和B都到达了synchronized 代码块前,A获得了锁,执行了实例的初次创建。A释放锁后,B获得锁,若不进行double check,实例将会重复创建。

java 复制代码
public class Singleton_02 {

    private static Singleton_02 instance;

    private Singleton_02() {
    }

    public static Singleton_02 getInstance(){
       if (null != instance) return instance;
       synchronized (Singleton_02.class){
         if (null != instance) return instance;
         instance = new Singleton_02();
         return instance;
         }
     }

}

饿汉模式(线程安全)

java 复制代码
public class Singleton_03 {

    private static Singleton_03 instance = new Singleton_03();

    private Singleton_03() {
    }

    public static Singleton_03 getInstance() {
        return instance;
    }

}

instance 在类加载时被创建,类加载过程是线程安全的。

使用类的内部类(线程安全)

java 复制代码
public class Singleton_04 {

    private static class SingletonHolder {
        private static Singleton_04 instance = new Singleton_04();
    }

    private Singleton_04() {
    }

    public static Singleton_04 getInstance() {
        return SingletonHolder.instance;
    }

}

既保证了线程安全又保证了懒加载,同时不会因为加锁的方式耗费性能。因为JVM虚拟机可以保证多线程并发访问的正确性,也就是一个类的构造方法在多线程环境下可以被正确的加载。

CAS(线程安全)

java 复制代码
public class Singleton_06 {

    private static final AtomicReference<Singleton_06> INSTANCE = new AtomicReference<Singleton_06>();

    private Singleton_06() {
    }

    public static final Singleton_06 getInstance() {
        for (; ; ) {
            Singleton_06 instance = INSTANCE.get();
            if (null != instance) return instance;
            INSTANCE.compareAndSet(null, new Singleton_06());
            return INSTANCE.get();
        }
    }

    public static void main(String[] args) {
        System.out.println(Singleton_06.getInstance()); // org.itstack.demo.design.Singleton_06@2b193f2d
        System.out.println(Singleton_06.getInstance()); // org.itstack.demo.design.Singleton_06@2b193f2d
    }

}
相关推荐
007php00711 分钟前
Go Vendor 和 Go Modules:管理和扩展依赖的最佳实践
java·开发语言·docker·微服务·golang·自动化·jenkins
郝学胜-神的一滴12 分钟前
C++组合模式:构建灵活的层次结构
开发语言·c++·程序人生·设计模式·组合模式
狂奔solar15 分钟前
使用Rag 命中用户feedback提升triage agent 准确率
java·前端·prompt
qianmoq18 分钟前
第07章:flatMap():处理嵌套数据的利器
java
AAA修煤气灶刘哥29 分钟前
后端人必懂的 “中间商” 哲学:代理模式 + 类加载器,从入门到唠明白
java·后端·面试
YANGZHAO31 分钟前
JavaCore:ArrayList源码解析与性能优化
java·后端
程序员水自流33 分钟前
Java设计模式是什么?核心设计原则有哪些?
java·设计模式
用户4130798106136 分钟前
面向对象六大设计原则
设计模式
YoungUpUp1 小时前
【电子设计自动化(EDA)】Altium Designer25——电子设计自动化(EDA)软件版保姆级下载安装详细图文教程(附安装包)
运维·设计模式·fpga开发·自动化·eda·电路仿真·电子设计自动化
用户3721574261351 小时前
使用 Java 替换和修改 PDF 文本的方法
java