【设计模式】Java设计模式-单例模式

【设计模式】Java设计模式 – 单例模式

😄 不断学习才是王道
🔥 继续踏上学习之路,学之分享笔记
👊 总有一天我也能像各位大佬一样
🌝分享学习心得,欢迎指正,大家一起学习成长!
原创作品,更多关注我CSDN: 一个有梦有戏的人
准备将博客园、CSDN一起记录分享自己的学习心得!!!

简介

单例模式,是java设计模式中最简单的设计模式,是属于创建类型模式。单例模式就是只能有一个实例,即一个类有且仅有一个实例,并且自行实例化向整个系统提供。

单例模式是设计模式中最简单的形式之一。这一模式的目的是使得类的一个对象成为系统中的唯一实例。要实现这一点,可以从客户端对其进行实例化开始。因此需要用一种只允许生成对象类的唯一实例的机制,”阻止”所有想要生成对象的访问。使用工厂方法来限制实例化过程。这个方法应该是静态方法(类方法),因为让类的实例去生成另一个唯一实例毫无意义。 — (美)钟冠贤.Objective-C编程之道 iOS设计模式解析.北京市:人民邮电出版社,2011

UML图:

【设计模式】Java设计模式-单例模式

以下使用多线程测试的代码(如以下例子):

class Threads extends Thread {
    @Override
    public void run() {
        StaticVariableStarve instance = StaticVariableStarve.getInstance();
        System.out.println("StaticVariableStarve hashCode: " + instance.hashCode());
    }
}

在main方法中调用start。

for (int i = 0; i < 5; i++) {
    new Threads().start();
}

1、饿汉式

①、饿汉静态变量

饿汉静态变量通过创建静态变量去实例化对象,在通过静态方法返回实例,其中需要实现私有化构造方法,使得外部不能通过new直接实例化对象。
代码如下:

package com.lyd.demo.singleton;

/**
 * @Author: lyd
 * @Description: 单例模式 - 饿汉静态变量
 * @Date: 2022-08-24
 */
public class StaticVariableStarve {
    // 私有化构造方法
    private StaticVariableStarve() {
    }
    // 创建静态变量实例化
    private final static StaticVariableStarve singleton = new StaticVariableStarve();
    // 返回对象
    public static StaticVariableStarve getInstance() {
        return singleton;
    }
}

饿汉静态变量的测试

单例模式创建只能有一个实例,不管是用几个变量,最后他们都是同一个实例,如下测试,通过”==”判断对象实例是否一致,打印其hashcode可以发现是一样的。

// 饿汉静态变量
StaticVariableStarve singleton1 = StaticVariableStarve.getInstance();
StaticVariableStarve singleton2 = StaticVariableStarve.getInstance();
System.out.println("两者是否相同?" + (singleton1 == singleton2));
System.out.println("singleton1的hashcode:" + (singleton1.hashCode()));
System.out.println("singleton2的hashcode:" + (singleton2.hashCode()));

运行结果:

【设计模式】Java设计模式-单例模式
多线程:
【设计模式】Java设计模式-单例模式

这样写在类装载的时候就已经实例化了,避免了线程安全问题。但是,一开始就已经实例化了,没有使用lazy loading,就会导致有时候这个实例不需要使用,但是他仍然实例化了,这样就造成了内存浪费。

②、饿汉静态代码块

静态代码块中实例化对象,通过类主动去实例化对象,当调用到这个类的时候,静态代码块的代码就会运行,从而实例化对象。
代码如下:

package com.lyd.demo.singleton;

/**
 * @Author: lyd
 * @Description: 饿汉静态代码块
 * @Date: 2022-08-24
 */
public class StaticBlockStarve {
    // 私有化构造方法
    private StaticBlockStarve() {
    }
    // 构建静态变量
    private static StaticBlockStarve singleton;
    static {
        singleton = new StaticBlockStarve();
    }
    // 返回对象
    public static StaticBlockStarve getInstance() {
        return singleton;
    }
}

饿汉静态代码块测试

// 饿汉静态代码块
StaticBlockStarve singleton3 = StaticBlockStarve.getInstance();
StaticBlockStarve singleton4 = StaticBlockStarve.getInstance();
System.out.println("两者是否相同?" + (singleton3 == singleton4));
System.out.println("singleton1的hashcode:" + (singleton3.hashCode()));
System.out.println("singleton2的hashcode:" + (singleton4.hashCode()));

运行结果:

【设计模式】Java设计模式-单例模式
多线程:
【设计模式】Java设计模式-单例模式

2、懒汉式

①、线程不安全

通过 Lazy 初始化,但是线程不安全,这种方式是最基本的实现方式,这种实现最大的问题就是不支持多线程。因为没有加锁 synchronized,所以严格意义上它并不算单例模式。
这种方式 lazy loading 很明显,不要求线程安全,在多线程不能正常工作。
代码如下:

通过getInstance()获取实例,在方法内进行判断是否为空,不为空直接返回对象实例。

package com.lyd.demo.singleton;

/**
 * @Author: lyd
 * @Description: 懒汉式-线程不安全
 * @Date: 2022-08-24
 */
public class NotSafeThreadLazy {
    // 私有变两个
    private static NotSafeThreadLazy singleton;
    // 私有构造
    private NotSafeThreadLazy() {
    }
    // 通过方法实例化
    public static NotSafeThreadLazy getInstance() {
        if (singleton == null) {
            singleton = new NotSafeThreadLazy();
        }
        return singleton;
    }
}

测试

// 懒汉线程不安全
NotSafeThreadLazy singleton1 = NotSafeThreadLazy.getInstance();
NotSafeThreadLazy singleton2 = NotSafeThreadLazy.getInstance();
System.out.println("两者是否相同?" + (singleton1 == singleton2));
System.out.println("singleton1的hashcode:" + (singleton1.hashCode()));
System.out.println("singleton2的hashcode:" + (singleton2.hashCode()));

运行结果

【设计模式】Java设计模式-单例模式
多线程:可见生成了多个实例
【设计模式】Java设计模式-单例模式

用到了lazy loading,但只能在单线程下使用。但是在多线程中,可能会导致一个线程已经到达了if判空,但是还没有进行实例化,第二个线程就已经进入判空,并且也进入实例化。这样就会破坏了单例模式,两个实例化就不是同一个,因此这种方式是线程不安全,不推荐使用。

②、线程安全

这种方式具备很好的 lazy loading,能够在多线程中很好的工作,但是,效率很低,99% 情况下不需要同步。
代码如下:

通过lazy loading实例化对象,只在第一次运行实例化,在getInstance方法中使用synchronized保证线程安全。

package com.lyd.demo.singleton;

/**
 * @Author: lyd
 * @Description: 懒汉式-线程安全
 * @Date: 2022-08-24
 */
public class SafeThreadLazy {
    // 私有变两个
    private static SafeThreadLazy singleton;
    // 私有构造
    private SafeThreadLazy() {
    }
    // 通过方法实例化
    public static synchronized SafeThreadLazy getInstance() {
        if (singleton == null) {
            singleton = new SafeThreadLazy();
        }
        return singleton;
    }
}

多线程 :

【设计模式】Java设计模式-单例模式

3、双检锁/双重校验锁(DCL,即 double-checked locking)

采用双锁机制,安全且在多线程情况下能保持高性能。

在变量加上关键字volatile,在实例化的时候用synchronized线程锁,可以使线程安全。
volatile:是Java虚拟机提供的轻量级的同步机制,当某线程更新变量后,其他线程也能感知到。

package com.lyd.demo.singleton;

/**
 * @Author: lyd
 * @Description: 双检锁/双重校验锁
 * @Date: 2022-08-24
 */
public class DoubleCheckedLocking {
    // 私有变两个
    private static volatile DoubleCheckedLocking singleton;
    // 私有构造
    private DoubleCheckedLocking() {
    }
    // 通过方法实例化
    public static DoubleCheckedLocking getInstance() {
        if (singleton == null) { // [1]
            synchronized (DoubleCheckedLocking.class) { // [2]
                if (singleton == null) {
                    singleton = new DoubleCheckedLocking();
                }
            }
        }
        return singleton;
    }
}

在getInstance()方法中,哪怕是多线程,都能保证线程安全以及单例模式。假如有多个线程,一个线程先到达了[1]判空成功后进入实例化,此时synchronized线程锁就会上锁,在还没实例化后,然而其他线程也过来了,他们也通过了[1]判空,但是在[2]处会被拦住,直到前一个实例化结束后才解锁,等到解锁后,也就已经实例化完成,singleton就已经不再是null。

多线程:

【设计模式】Java设计模式-单例模式

4、静态内部类

这种方式能达到双检锁方式一样的功效,但实现更简单。对静态域使用延迟初始化,应使用这种方式而不是双检锁方式。这种方式只适用于静态域的情况,双检锁方式可在实例域需要延迟初始化时使用。
代码如下:

通过SingletonHolder这个静态内部类实例化,采用类装载的方式进行实例化,但是一开始是不会实例化的,只有显式调用了getInstance(),使得SingletonHolder类被主动调用,从而实例化对象。

package com.lyd.demo.singleton;

/**
 * @Author: lyd
 * @Description: 静态内部类
 * @Date: 2022-08-24
 */
public class StaticInnerClass {
    private StaticInnerClass() {}
    // 静态内部类
    public static class SingletonHolder {
        private static final StaticInnerClass SINGLETON = new StaticInnerClass();
    }
    // 调用方法
    public static StaticInnerClass getInstance() {
        return SingletonHolder.SINGLETON;
    }
}

采用了类装载的方式使实例化只有一个线程。在一开始StaticInnerClass被装载的时候,对象不一定被实例化。需要调用了getInstace()的方法,才会使得SingletonHolder被主动使用,装载SingletonHolder类,也就接着进行实例化对象。

多线程:

【设计模式】Java设计模式-单例模式

5、枚举

这种实现方式还没有被广泛采用,但这是实现单例模式的最佳方法。它更简洁,自动支持序列化机制,绝对防止多次实例化。
这种方式是 Effective Java 作者 Josh Bloch 提倡的方式,它不仅能避免多线程同步问题,而且还自动支持序列化机制,防止反序列化重新创建新的对象,绝对防止多次实例化。
代码如下:

package com.lyd.demo.singleton;

/**
 * @Author: lyd
 * @Description:
 * @Date: 2022-08-24
 */
public enum Enumeration {
    INSTANCE;
    public void instanceMethod() {
        System.out.println("实例的方法");
    }
}

测试

// 枚举
Enumeration instance = Enumeration.INSTANCE;
Enumeration instance2 = Enumeration.INSTANCE;
System.out.println("两者是否相同?" + (instance == instance2));
System.out.println("instance的hashcode:" + (instance.hashCode()));
System.out.println("instance2的hashcode:" + (instance2.hashCode()));
instance.instanceMethod();

结果

【设计模式】Java设计模式-单例模式
多线程测试:
【设计模式】Java设计模式-单例模式
可发现多线程下枚举也是能够实现单例模式。

👍创作不易,如有错误请指正,感谢观看!记得一键三连哦!👍

💓德德小建议:

理解设计模式不是一件简单的事情,需要不断的学习和动手去练习,才能理解。只有掌握好设计模式,才能够真正的理解SpringAOP和Mybatis的底层原理。各位读者可以和我一样,动手敲一敲代码,甚至用不同的例子来做,通过debug一步一步调试,还有就是多看看别人的例子。能够有助于理解!谢谢各位观看指点!❤️ ❤️ ❤️

Original: https://www.cnblogs.com/lyd-code/p/16661448.html
Author: 怒放吧德德
Title: 【设计模式】Java设计模式-单例模式

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/574138/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • GCC常见命令

    rwx 对于目录和文件的区别 文件 目录 r 文件的内容可以被查看。支持cat、more、head…vim 目录的内容可以被查看。ls、tree w 文件的内容可以被添…

    Linux 2023年6月6日
    0115
  • 设计模式——创建型设计模式

    创建型设计模式 争对 &#x5BF9;&#x8C61;/&#x7C7B;创建时的优化 工厂方法模式(了解) 通过定义顶层抽象工厂类,通过继承的方式,针对于每…

    Linux 2023年6月7日
    097
  • Debian中CodeIgniter+nginx+MariaDB+phpMyAdmin配置

    本文不讲述软件安装过程,记述本人在Debia 中配置CodeIgniter 时遇到的问题及解决方法,希望能够为有需要的人提供帮助。 一、Debian版本及所需的软件 Debian …

    Linux 2023年6月13日
    099
  • VS 2010 LINK fatal error LNK1123转换到 COFF 期间失败 文件无效或损坏

    1. 解决LINK : fatal error LNK1123: 转换到 COFF 期间失败: 文件无效或损坏 VS 2010 【LINK : fatal error LNK112…

    Linux 2023年6月13日
    082
  • 面试必问的安卓虚拟机,你真的掌握了么?——安卓虚拟机基础知识回顾

    前言 21世纪,安卓虚拟机正在一步步的走入我们的生活,小到个人部分朋友在电脑上使用安卓虚拟机玩手游,大到安卓从业人员在虚拟机上面跑程序。不得不承认,对于每一位Androider 而…

    Linux 2023年6月13日
    097
  • CentOS 7.x 用shell增加、删除端口

    一、在/usr/local/sbin/下创建port文件,不要扩展名,并给权限 chom 777 port 二、用法 #port add 8080 #port remove 808…

    Linux 2023年5月28日
    098
  • 018.CentOS升级内核

    作者:木二 出处:http://www.cnblogs.com/itzgr/ 关于作者:云计算、虚拟化,Linux,多多交流! 本文版权归作者所有,欢迎转载,但未经作者同意必须保留…

    Linux 2023年6月13日
    0117
  • Redis AOF重写

    AOF 持久化是通过保存被执行的写命令来记录数据库状态的,所以AOF文件的大小随着时间的流逝一定会越来越大;影响包括但不限于:对于Redis服务器,计算机的存储压力;AOF还原出数…

    Linux 2023年5月28日
    0102
  • 常用命令-watch

    每隔一秒高亮显示网络链接数的变化情况 每隔一秒高亮显示http链接数的变化情况 实时查看模拟攻击客户机建立起来的连接数 监测当前目录中 scf 的文件的变化 10秒一次输出系统的平…

    Linux 2023年6月6日
    0118
  • Centos7升级内核

    1、查看当前内核版本 $ uname -r 3.10.0-1160.25.1.el7.x86_64 $ uname -a Linux localhost.localdomain 3…

    Linux 2023年6月13日
    074
  • vue-admin-template组件前端,登录验证成功后,getInfo方法不调用

    先查看request.js中自定义状态码是否为自己的定义的成功状态码 这里的状态码该为自己的成功状态码 Original: https://www.cnblogs.com/antl…

    Linux 2023年6月7日
    0108
  • 软件工程 软件需求与软件需求规约 第1篇随笔

    2、软件需求与软件需求规约 1. 何为需求? 定义问题的基本要素是 “需求” 一个需求是一个有关”要予构造”的陈述,用以描述待开发产…

    Linux 2023年6月7日
    0100
  • [Python]Tkinter 做简单的窗口视窗GUI(参考莫烦笔记)

    Label & Button 标签与按钮 Entry & Text 输入与文本框 ListBox 列表部件 Radiobutton 选择按钮 Scale 尺度 Ch…

    Linux 2023年6月13日
    0128
  • Linux(进阶篇)

    一、进程 1 进程和内存管理 1.1 进程和线程的区别 线程是程序执行的最小单位,而进程是操作系统分配资源的最小单位 一个进程由一个或多个线程组成,线程是一个进程中代码的不同执行路…

    Linux 2023年6月7日
    090
  • Git简介

    Git是一个开源的分布式版本控制系统,是目前主流的版本控制系统,很多软件项目都会用它做源代码管理。Git的常用操作想必很多人都会,但是可能了解Git内部原理的人并不多。了解一些底层…

    Linux 2023年6月6日
    079
  • 生成符合chrome要求的自签名HTTPS证书

    按照文章给Nginx配置一个自签名的SSL证书中给出的代码生成自签名证书后,发现使用chrome浏览器访问会报 Invalid self signed SSL cert &#821…

    Linux 2023年6月6日
    077
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球