Java--设计模式之单例模式+保证线程安全的5种写法(懒、汉、内、双、枚)

news/2024/5/17 15:48:25 标签: java, 单例模式, 设计模式, volatile, synchronized
❤️‍您好,我是贾斯汀,今天来聊一聊关于单例模式!❤️‍

什么是设计模式

百科:

设计模式是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。

设计模式是软件行业的通用的设计标准,在Java同样通用,主要有23种设计模式如下:
在这里插入图片描述
有的小伙伴可能会问,这么多,学得完吗?

答:不好意思,不要太自信了,一般人还真学不完,不过一些常用的设计模式,例如上图中标红的单例模式工厂模式代理模式设计模式,还是需要花些时间和精力去多多了解一下,相信会对自己在程序设计或写代码时有很大的帮助。

本文主要来聊一聊设计模式中创建型的单例模式,进入正文~

单例模式是什么?

学习Java的小伙伴,相信都写过Class类吧,创建某个类实例化对象的核心是new MyClass()来实现,如果没有任何设计规范,在日常开发写代码时,如果实例被用的地方很多,每次调用的时候都通过new MyClass()得到实例化对象,代码重复而且频繁的创建对象还影响性能,而有些场景我们只需要提供该类的一个实例即可,例如平时比较常见的线程池、日志对象、缓存等,一般只需要确保有一个实例即可,这种确保某个类只有一个实例并且能够类自身提供自动创建实例化对象的设计模式即称为单例模式

单例模式设计的原则是什么?

  • 构造方法私有化:既然是单例,就不能将类的构造函数暴露在外面,因此需要重写构造函数为私有化;
  • 要考虑线程安全:多线程环境下,要确保不会构造出多个实例对象。

Java实现单例模式的5种方式?

关于Java实现单例模式的有几种方式,网上有很多说法,有5种、6种甚至7种实现方式,本文出于单例模式设计的两个主要原则构造方法私有化要考虑线程安全,不考虑线程安全的其他实现方式没有任何意义,主要有5种实现方式:
在这里插入图片描述

懒汉

使用懒汉式写法,主要是通过synchronized修饰实例化方法getInstance,保证了线程安全,并且只有调用getInstance时才初始化,顾此得名懒汉。
懒汉写法1:

java">/**
 * 单例模式之懒汉写法1
 */
public class Singleton {
    private static Singleton instance = null;
    private Singleton(){}
    public synchronized static Singleton getInstance(){
        if (instance == null){
            instance = new Singleton();
        }
        return instance;
    }
}

懒汉写法2:
该写法等价于写法1,原因在于关键字synchronized的灵活运用,放在方法上修饰,加锁的对象是Singleton,等效于将synchronized移到方法内部作为一个同步块,并通过括号中的Singleton.class显示指定锁对象,效果是一样的。

java">/**
 * 单例模式之懒汉写法2
 */
public class Singleton {
    private static Singleton instance = null;
    private Singleton(){}
    public static Singleton getInstance(){
        synchronized(Singleton.class) {
            if (instance == null) {
                instance = new Singleton();
            }
        }
        return instance;
    }
}

饿汉

饿汉写法,只需要定义一个static静态变量instance = new Singleton(),简单的理解为在类加载时,也会完成单例对象的实例化工作。

java">/**
 * 单例模式之饿汉
 */
public class Singleton {
    private static Singleton instance = new Singleton();
    private Singleton(){}
    public static Singleton getInstance(){
        return instance;
    }
}

细心的小伙伴会发现该过程并没有使用到synchronized关键字,那会不会线程不安全呢?答案是,不会,如果你大概了解过Java虚拟机即JVM(Java Virtual Machine),那你可能知道类加载过程为:加载 -> 验证 ->解析 ->初始化,而初始化阶段是执行类构造器<clinit>()方法的过程,<clinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块中的语句合成产生的。
《深入理解Java虚拟机》类加载机制章节部分说明:

虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确地加锁、同步,如果多个线程同时如初始化一个类,那么只会有一个线程去执行这个类的<clinit>()方法,其他线程都需要阻塞等待,知道活动线程执行<clinit>()方法完毕。

静态内部类

静态内部类这种方式,其实就是在类的内部创建一个static SingletonInner静态内部类,然后在静态内部类的内部再定义一个static final修饰的静态常量INSTANCE = new Singleton(),同样static修饰的SingletonInner静态内部类,会在JVM加载类时完成类的初始化并完成自己定义的静态常量单例实例化过程。

java">/**
 * 单例模式之静态内部类
 */
public class Singleton {
    private static class SingletonInner{
        private static final Singleton INSTANCE = new Singleton();
    }
    private Singleton(){}
    public static Singleton getInstance(){
        return SingletonInner.INSTANCE;
    }
}

双重校验锁DCL(Double Check Lock)

DCL写法,其实与单例模式之懒汉写法2区别在于,synchronized同步块外面再套一层判断,并且使用了能确保线程安全核心volatile关键字修饰instance,表明单例变量是内存共享的,能够保证在多线程环境下的即时可见性

java">/**
 * 单例模式之双重校验锁DCL
 */
public class Singleton {
    private volatile static Singleton instance = null;
    private Singleton(){}
    public static Singleton getInstance(){
        if ( instance  == null ){
            synchronized (Singleton.class){
                if (instance  == null)
                    instance = new Singleton();
            }
        }
        return instance;
    }
}

枚举(num)

枚举方式很容易被大家给忽略掉了,但这种方式我觉得是最简单且又友好的一种推荐创建单例的方式,通过enum修饰Singleton单例类,仅需定义一个INSTANCE,然后在静态方法实例化方法getInstance中直接返回INSTANCE即可。

java">/**
 * 单例模式之枚举
 */
public enum Singleton {
    INSTANCE;
    public static Singleton getInstance(){
        return INSTANCE;
    }
}

小结

设计模式单例模式,看似挺简单,其实还涉及了枚举enum、同步锁synchronized、JVM类加载机制、多线程volatile关键字的使用等Java的N个知识点。

本文提到的单例模式懒饿内双枚5种方式,你学废了吗?

最后,学完希望你能熟悉的手写出任意一种实现单例模式的方式,并且对每一种写法是如何保证线程安全的原理也能够略知三四,祝你学习进步、工作顺利。


http://www.niftyadmin.cn/n/1328195.html

相关文章

LeetCode--Java实现704.二分查找、278.第一个错误的版本、35.搜索插入位置

学习背景 本文主要介绍如何通过Java实现LeetCode官方提供的以下数据结构与算法题目&#xff1a; 704.二分查找278.第一个错误的版本35.搜索插入位置 目录学习背景704.二分查找题目分析解题思路代码实现278.第一个错误的版本题目分析解题思路代码实现35.搜索插入位置题目分析解题…

MySQL--新手必备SQL基础知识、事务ACID及隔离级别

❤️‍您好&#xff0c;我是贾斯汀&#xff0c;本文主要分享数据库的一些基础知识&#xff01;❤️‍SQL 什么是SQL&#xff1f; 【百度百科】 结构化查询语言&#xff08;Structured Query Language&#xff09;简称SQL&#xff0c;是一种特殊目的的编程语言&#xff0c;是一…

LeetCode--只出现一次的数字(位运算、Set集合)

题目分析 原题&#xff1a; 给定一个非空整数数组&#xff0c;除了某个元素只出现一次以外&#xff0c;其余每个元素均出现两次。找出那个只出现了一次的元素。 说明&#xff1a;你的算法应该具有线性时间复杂度。 你可以不使用额外空间来实现吗&#xff1f; 示例 1: 输入: […

js里面怎样将一个边框的高度随着一个添加函数的执行而增加_你怎样理解前端的工作?...

作者&#xff1a;李文杨来源&#xff1a; cnblogs.com/Smiled/p/8377188.html入坑前端到今天也将近两年半了&#xff0c;这两天突然想到了第一次面试时面试官的一个问题&#xff1a;你怎样理解前端的工作&#xff1f;对于当时我一个小白而言完全是胡说一通&#xff0c;词不达意…

LeetCode--多数元素(数组排序、Map特性、位运算、摩尔投票)

题目分析 原题&#xff1a; 给定一个大小为 n 的数组&#xff0c;找到其中的多数元素。多数元素是指在数组中出现次数 大于 ⌊ n/2 ⌋ 的元素。 你可以假设数组是非空的&#xff0c;并且给定的数组总是存在多数元素。 进阶&#xff1a; 尝试设计时间复杂度为 O(n)、空间复杂度…

Linux--防火墙iptables基本命令、常用端口的开放/阻止/删除

【学习背景】​ Linux CentOS 6.5版本以前&#xff0c;默认防的火墙是iptables&#xff0c;CentOS6.5版本及以后版本&#xff0c;防火墙都由iptables升级为了firewall&#xff0c;不过底层还是基于iptables的指令&#xff0c;因此还是有必要了解了解。 本文主要介绍iptables的基…

densenet提取特征_CondenseNet:可学习分组卷积,原作对DenseNet的轻量化改进 | CVPR 2018...

CondenseNet特点在于可学习分组卷积的提出&#xff0c;结合训练过程进行剪枝&#xff0c;不仅能准确地剪枝&#xff0c;还能继续训练&#xff0c;使网络权重更平滑&#xff0c;是个很不错的工作来源&#xff1a;晓飞的算法工程笔记 公众号论文:Neural Architecture Search with…

MySQL--关于my.cnf配置文件中的常见参数、参数值及参数说明

【学习背景】 本文主要分享一下MySQL日常开发运维当中&#xff0c;关于配置文件my.cnf中[client]、[mysqld]、[mysql]、[mysqld_safe]四个组下比较常见的参数、参数值以及参数值说明。 学习目录一、[client]组下参数二、[mysqld]组下参数三、[mysql]组下参数四、[mysqld_safe]组…