IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Java常用设计模式—单例模式 -> 正文阅读

[Java知识库]Java常用设计模式—单例模式

GoF23 (二十三种设计模式)可按照:创建型模式、结构性模式、行为型模式三类

创建型模式

  • 单例模式、工厂模式、抽象工厂模式、建造者模式、原型模式

结构性模式

  • 适配器模式、桥接模式、装饰模式、外观模式、享元模式、代理模式

行为型模式

  • 模板方法模式、命令模式、迭代器模式、观察者模式、中介者模式、备忘录模式、解释器模式、状态模式、策略模式、职责链模式、访问者模式

初学设计模式:让我们从单例模式开始:

一、单例模式

单例模式使?场景:

  • 业务系统全局只需要?个对象实例,?如发号器、 redis 连接对象等
  • Spring IOC容器中的 bean 默认就是单例
  • spring boot 中的controller、service、dao层中通过 @Autowired的依赖注?对象默认都是单例的

单例模式分类:

  • 懒汉:就是所谓的懒加载,延迟创建对象,需要用的时候再创建对象
  • 饿汉:与懒汉相反,提前创建对象

1、饿汉式单例

类加载的时候直接创建对象,不管对象是否会用到!

package com.sqx.gof.single;

/***
 * 饿汉式单例,类加载的时候直接创建对象,不管对象是否会用到!
 *
 * 存在问题:对象如果没有被使用那么,其中data1~4 会占用我们的内存!
 */

public class HungrySingle {

    private byte[] data1 = new byte[1024*1024] ;
    private byte[] data2 = new byte[1024*1024] ;
    private byte[] data3 = new byte[1024*1024] ;
    private byte[] data4 = new byte[1024*1024] ;

    private HungrySingle(){   //私有构造,只能再类的内部使用,外部不可通过new创造对象

    }

    private static HungrySingle  HUNGRY = new HungrySingle();  //类加载的时候执行

    public static HungrySingle getInstance(){
        return HUNGRY ;
    }
}

2、懒汉式单例

需要获取对象的时候,才去创建对象!

package com.sqx.gof.single;

/***
 * 懒汉式单例
 */
@SuppressWarnings("ALL")
public class LazySingle {

    private LazySingle(){  //私有化构造
        System.out.println("当前线程名称:" + Thread.currentThread().getName());
    }

    private static LazySingle Lazy ;

    public static LazySingle getInstance(){  //调用该方法获取对象的时候,才去创建对象!
        if (Lazy == null) {
              Lazy = new LazySingle();
    }
            return Lazy ;
    }

    public static void main(String[] args) {
        for (int i = 0; i < 10 ; i++) {
            new Thread(() -> {
                getInstance() ;           //创建十个线程去测试!
            }).start();
        }
    }
}

发现上述代码存在问题如下:

在这里插入图片描述

我们发现在多线程的情况下,会发生线程安全问题

需要加锁,保证我们的线程安全问题 ,这里我们使用的是DCL(double check Lock)双检锁

    public static LazySingle getInstance(){  //调用该方法获取对象的时候,才去创建对象!
        if (Lazy == null) {
            synchronized (LazySingle.class){
                if (Lazy == null){
                    Lazy = new LazySingle();
                }
            }
    }
            return Lazy ;
}

在这里插入图片描述

看着结果是没问题了,但是在多线程情况下,由于new对象不是原子性操作,上述代码仍然存在指令重排的问题

我们先看一下new 的字节码指令:

// 通过这个复制的引用调用它的构造方法
21: invokespecial #4 // Method "<init>":()V
// 最开始的这个引用用来进行赋值操作
24: putstatic #2 // Field INSTANCE:Lcn/itcast/n5/Singleton;

此时我们的发生指令重排,先执行赋值操作,将空的实例对象返回(此时Lazy实例已经有值了),然后执行构造初始化对象!

就在我们的的初始化执行一半,线程t2过来了,发现Lazy不为null,执行 return Lazy;我们此时返回还是未被初始化的对象,所

以问题就此发生!!

通过volatile解决指令重排问题即可:

   private static volatile LazySingle Lazy ;

对volatile修饰的变量进行写操作的时候,在写操作后加上内存屏障,使得写屏障之前的代码不会发生指令重排!

小结

饿汉式单例模式,当类加载的时候就直接实例化对象,因此不需要考虑线程安全问题

  • 优点:实现简单,不需要考虑线程安全问题
  • 缺点:不管有没有使用该对象实例,instance对象一直占用着这段内存

懒汉与饿汉式如何选择

  • 如果对象内存占用不大,且创建不复杂,直接使用饿汉的方式即可
  • 其他情况均采用懒汉方式(优选)
  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-01-24 10:41:48  更:2022-01-24 10:42:22 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 9:52:24-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码