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

不断学习才是王道

继续踏上学习之路,学之分享笔记

总有一天我也能像各位大佬一样

分享学习心得,欢迎指正,大家一起学习成长!

原创作品,更多关注我CSDN: 一个有梦有戏的人

准备将博客园、CSDN一起记录分享自己的学习心得!!!

简介

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

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

UML图:

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

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()));

运行结果:



多线程:

这样写在类装载的时候就已经实例化了,避免了线程安全问题。但是,一开始就已经实例化了,没有使用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()));

运行结果:



多线程:

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()));

运行结果



多线程:可见生成了多个实例

用到了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;
}
}

多线程 :

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。

多线程:

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类,也就接着进行实例化对象。

多线程:

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();

结果



多线程测试:



可发现多线程下枚举也是能够实现单例模式。

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

德德小建议:

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

【设计模式】Java设计模式 - 单例模式的更多相关文章

  1. Unity设计模式+Java设计模式,讲解+案例+PPT,一次性学会设计模式,拥抱高薪!

    一个程序员对设计模式的理解:“不懂”为什么要把很简单的东西搞得那么复杂.后来随着软件开发经验的增加才开始明白我所看到的“复杂”恰恰就是设计模式的精髓所在,我所理解的“简单”就是一把钥匙开一把锁的模式, ...

  2. 设计模式 - Java中单例模式的6种写法及优缺点对比

    目录 1 为什么要用单例模式 1.1 什么是单例模式 1.2 单例模式的思路和优势 2 写法① - 饥饿模式 2.1 代码示例 2.2 优缺点比较 3 写法② - 懒惰模式 3.1 代码示例 3.2 ...

  3. 简单工厂设计模式--Java设计模式(一)

    一 概念: 简单工厂模式就是通过一个工厂类根据参数动态创建对应的类. 二 案例 我们以汽车作为例子,在现实生活中汽车只是一个抽象的产品,其中有很多类型的汽车才是具体产品,如奔驰.宝马.保时捷等等(当然 ...

  4. Java设计模式学习资源汇总

    本文记录了Java设计模式学习书籍.教程资源.此分享会持续更新: 1. 设计模式书籍 在豆瓣上搜索了一把,发现设计模式贯穿了人类生活的方方面面.还是回到Java与程序设计来吧. 打算先归类,再浏览,从 ...

  5. Java设计模式知识整理

    1.Java设计模式     Java设计模式分为三种类型,分别是:      ①.创建型设计模式:是对对象创建过程的各种问题和解决方案的总结           包括:静态工厂模式.抽象工厂模式.单 ...

  6. Java设计模式学习笔记(一) 设计模式概述

    前言 大约在一年前学习过一段时间的设计模式,但是当时自己的学习方式比较低效,也没有深刻的去理解.运用所学的知识. 所以现在准备系统的再重新学习一遍,写一个关于设计模式的系列博客. 废话不多说,正文开始 ...

  7. 折腾Java设计模式之中介者模式

    博文原址:折腾Java设计模式之中介者模式 中介者模式 中介者模式(Mediator Pattern)是用来降低多个对象和类之间的通信复杂性.这种模式提供了一个中介类,该类通常处理不同类之间的通信,并 ...

  8. java设计模式——建造者模式

    一. 定义与类型 定义:将一个复杂对象的构建与它的表示分离,使用同样的构建过程可以创建不同的表示 用户只需制定需要建造的类型就可以得到它们,建造过程以及细节不需要知道 类型:创建型 建造者模式与工厂模 ...

  9. java设计模式之单例模式(几种写法及比较)

    概念: Java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建 ...

  10. Java 设计模式 —— 单例模式

    1. 概念: 单例模式是一种常用的软件设计模式.核心结构中只包含一个被称为单例的特殊类.通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源.如果 ...

随机推荐

  1. FS2K人脸素描属性识别

    人脸素描属性识别 代码:https://github.com/linkcao/FS2K_extract 问题分析 需要根据FS2K数据集进行训练和测试,实现输入一张图片,输出该图片的属性特征信息,提取 ...

  2. PotPlayer播放百度云盘视频

    需要的工具 PotPlayer.油猴tampermonkey.坚果(这个不用下载,有个账号就行) 下载地址:百度网盘 步骤 安装油猴tampermonkey 拖拽Tampermonkey_4.14.c ...

  3. Python爬虫常用:谷歌浏览器驱动——Chromedriver 插件安装教程

    我们在做爬虫的时候经常要使用谷歌浏览器驱动,今天分享下这个Chromedriver 插件的安装方法. 第一步:打开谷歌浏览器打开设置面板 嫌枯燥的小伙伴可以点击此处找管理员小姐姐领取免费资料 第二步: ...

  4. JAVA编程练习01作业

    1.已知y与x的关系:,要求:从键盘上输入一个x的值,输出其对应的y的值. 2. 输入一个圆半径(r),计算并输出圆的面积和周长. 3.输入一个三位正整数n,输出其个位.十位和百位上的数字. 4.根据 ...

  5. 使用xpath查找元素的子元素,找不到

    List<WebElement> allUserList = driver.findElements(By.xpath("xxx")); for (WebElement ...

  6. .net webapi 实现 接口版本控制并打通swagger支持

    我们在开发 webapi 项目时如果遇到 api 接口需要同时支持多个版本的时候,比如接口修改了入参之后但是又希望支持老版本的前端(这里的前端可能是网页,可能是app,小程序 等等)进行调用,这种情况 ...

  7. postgresql自增id

    drop index Ix_product_define_id; drop index Ix_user_umid; drop table invims_product_attention; /*=== ...

  8. 从编译器对指令集的要求看API设计原则

    摘要:最近看<计算机体系结构:量化研究方法(第五版)>,发现指令集设计中的一些原则,对API设计也同样适用,给大家分享一下. 本文中的所有内容来自工作和学习过程中的心得整理,如需转载请注明 ...

  9. Webpack干货系列 | 怎么运用 Webpack 5 处理css/scss/sass、less、stylus样式资源

    程序员优雅哥简介:十年程序员,呆过央企外企私企,做过前端后端架构.分享vue.Java等前后端技术和架构. 本文摘要:主要讲解webpack 5 如何高效处理CSS 资源.scss/sass 资源.l ...

  10. 基于yarn1.x的monorepo实践分享

    背景介绍 几天前,晓东船长微信问我,你们团队有没有monorepo的实践,我很遗憾的告诉他没有,但这在我心里播下了一颗探索的种子,刚好最近老总要搞内蒙古的新项目,我和另一个前端兄弟组成双枪敢死队进行保 ...