Java创建对象的最佳方式:单例模式(Singleton)

前言

单例模式是java中最简单的设计模式之一,属于创建式模式,提供了一种创建对象的最佳方式。

具体而言,单例模式涉及到一个具体的类,这个类可以确保只有单个对象被创建。它包含一个访问其唯一对象的方法,供外部直接调用,而不需要创建这个类的示例。

简而言之,可以不再new一个他的实例,而是直接调用方法。

实现

单例模式分为两种:

  • 饿汉式:类加载时就会导致该单例对象被创建
  • 懒汉式:类加载不会导致该单例对象被创建,而是首次使用时才会

饿汉式

静态变量方式

public class Singleton {
    // 私有构造方法
    private Singleton() {}
    
    // 静态方式创建该类的对象
    private static Singleton instance = new Singleton();

    // 对外提供静态方法,用于获取该对象
    public static Singleton getInstance() {
        return instance;
    }
}

将对象的实例创建为静态,保证了实例永远只有一份,且在该类加载时就会立即创建在jvm内存的方法区,程序运行期间永久存在,因此当对象太大时会造成浪费。

静态代码块方式

public class Singleton {
    // 私有构造方法
    private Singleton() {}
    
    // 静态创建该类对象
    private static Singleton instance;
    static {
        instance = new Singleton();
    }

    // 对外提供静态方法,用于获取该对象
    public static Singleton getInstance() {
        return instance;
    }
}

对象的创建在代码块中,也是随着类的加载而创建,两种方式基本相同。

 懒汉式

从名字出发,饿汉式会一直饿着,需要不断有食物,即对象一直存在。

懒汉式则是只有在饿的时候才会寻找食物,即请求对象实例。

所以,只要将饿汉式的创建对象放到getInstance方法中,即只有在调用该方法时才创建,我们就完成了懒加载的效果。

更改初始化时机

public class Singleton {
    // 私有构造方法
    private Singleton() {}
    
    // 静态创建该类对象
    private static Singleton instance;

    // 对外提供静态方法,用于获取该对象
    public static Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

但这样的代码是线程不安全的,可以设想,如果有两个线程同时调用getInstance方法,他们都发现了instance == null,那么此时就会有两个实例化的instance,这就违背了单例模式的初衷了。

 最容易想到的方法就是给getInstance方法上锁。

上锁:防止初始化多次

public class Singleton {
    // 私有构造方法
    private Singleton() {}
    
    // 静态创建该类对象
    private static Singleton instance;

    // 上锁
    public static synchronized Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

但此处依然有优化的空间,如果instance不为空,让线程调用就好了,不一定要持有锁,因此我们引申出双重检查锁模式。

优化:双重锁模式

public class Singleton {
    // 私有构造方法
    private Singleton() {}
    
    // 静态创建该类对象
    private static Singleton instance;

    // 上锁
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                // 线程拥有锁之后,再判断是否为空
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

为什么要两次判断是否为空?

例如有ab两个线程,当a先获得锁并初始化instance实例后,b获得锁,如果此时不判空,则会重复初始化,因为此时两者都进入了锁之内了,上一个判空已经过时。

但是,jvm在实例化对象的时候,会进行优化和指令重排序操作,多线程的情况下,可能会产生空指针,此时我们可以使用volatile关键字。

使用volatile关键字修饰的变量,可以保证其指令执行的顺序与我们写明的顺序一致,最终代码如下:

public class Singleton {
    // 私有构造方法
    private Singleton() {}
    
    // 静态创建该类对象
    private static volatile Singleton instance;

    // 上锁
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                // 线程拥有锁之后,再判断是否为空
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

最终进化:枚举实现

在java1.5之后,使用java实现单例模式的方式多了一种枚举。

public enum Singleton {
    INSTANCE;

    public void doSomething() {
        System.out.println("这是枚举类型的单例模式!");
    }
}

我们可以对枚举方式实现的实例进行反编译,可以得到如下的代码:

public final class Singleton extends Enum {
    public static Singleton[] values() {
        return (Singleton[])$VALUES.clone();
    }
    
    public static Singleton valueOf(String name) {
        return (Singleton)Enum.valueOf(com/qgn/mianshi/main/Singleton, name);
    }
    
    private Singleton(String s, int i) {
        super(s, i);
    }
        
    public static final Singleton INSTANCE;
    
    private static final Singleton $VALUES[];
    
    static {
        INSTANCE = new Singleton("INSTANCE", 0);
        $VALUES = (new Singleton[] {
            INSTANCE
        });
    }
}

可见Singleton继承了Enum,并重写了Enum类中的一些方法,基本上这些方法都是静态的。

这种实现方式的优点在于,不需要做额外的操作去保证对象单一性与线程安全,同时使用枚举可以防止调用者使用反射、反序列化强制生成多个单例对象,破坏单例模式。

但是如果没使用枚举时,要怎么防止反射和反序列化破坏单例呢?

防止反射、反序列化破坏单例模式

防反射

反射是一种暴力获取对象实例的方法,可以直接获取private修饰的构造函数,所以在放反射上只能被动防御。

既然能访问构造函数,那我们也可以在构造函数中建立防御机制。

public class Singleton  {
    private static volatile boolean flag = false;
    //私有构造方法
    private Singleton() {
        synchronized (Singleton.class){
        if (flag){
                throw new RuntimeException();
        }
         flag = true;
    }}
    private static class SingletonHolder {

        private static final Singleton INSTANCE = new Singleton();
    }
    //对外提供静态方法获取该对象
    public static Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
}

防反序列化

在readObject()方法的调用栈的底层方法中有这么两个方法:
hasReadResolveMethod:
表示如果实现了serializable 或者 externalizable接口的类中包含readResolve则返回true。

所以,原理也就清楚了,主要在Singleton中定义readResolve方法,并在该方法中指定要返回的对象的生成策略,就可以防止单例被破坏。

public class Singleton implements Serializable {
    //私有构造方法
    private Singleton() {}
    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }
    //对外提供静态方法获取该对象
    public static Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
    private Object readResolve() {
        return SingletonHolder.INSTANCE;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/581284.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

网页模版如何用

现在的网页模版已经得到了许多人的喜爱和使用。随着人们对互联网的需求不断增加,更多的公司和组织需要拥有自己的网站,以推广他们的品牌和服务。而网页模版为他们提供了一个简单而高效的方法来创建自己的网站。 网页模版是预先设计好的网站模板&#xff…

【数据分析】NumPy

文章目录 [toc]ndarray的创建np.array()方法np.arange()方法np.zeros()方法np.ones()方法np.full()方法np.eye()方法np.random模块np.random.random()方法np.random.randint()方法np.random.choice()方法np.random.shuffle()方法 ndarray的属性ndarray.dtypendarray.ndimndarra…

初识BootStrap

目录 前言: 1.Bootstrap的特点包括: 1.1响应式设计: 1.2组件丰富: 1.3易于定制: 1.4兼容性良好: 1.5强大的社区支持: 1.6一致的样式和布局: 1.7 插件和扩展性 2.初识Ajax: 2.1同步请求…

Linux——(关于权限常见的3个问题)

文章目录 1.修改文件或者目录的拥有者和所属组1.1chown指令1.2chgrp指令 2.常见的权限三个问题2.1对应一个目录,如果要进入,需要什么权限?2.2为什么我们创建的文件默认权限不是7772.2.1关于Linux下的权限掩码 2.3文件能否被删除取决于什么2.3…

Paddle OCR v4 微调训练文字识别SVTRNet模型实践

文字识别步骤参考:https://github.com/PaddlePaddle/PaddleOCR/blob/main/doc/doc_ch/recognition.md 微调步骤参考:https://github.com/PaddlePaddle/PaddleOCR/blob/release/2.7.1/doc/doc_ch/finetune.md 训练必要性 原始模型标点符号和括号容易识别不到 数据…

【kettle004】kettle访问本地MySQL数据库并处理数据至execl文件

一直以来想写下基于kettle的系列文章,作为较火的数据ETL工具,也是日常项目开发中常用的一款工具,最近刚好挤时间梳理、总结下这块儿的知识体系。 熟悉、梳理、总结下MySQL关系数据库相关知识体系 3.欢迎批评指正,跪谢一键三连&…

大模型微调:技术迭代与实践指南

在人工智能领域,大模型(LLM)的微调是一个关键过程,它使模型能够适应特定的任务和数据集。微调是深度学习中用于改进预训练模型性能的重要技术。通过在特定任务的数据集上继续训练,模型的权重被更新以更好地适应该任务。…

揭秘工业大模型:从人工智能小白到技术先锋

工业大模型的五个基本问题 信息化时代,数字化转型成为企业提升营运效率、应对经营风险和提升核心竞争力的重要途径。在此过程中,数据作为一种客观存在的资源,所产生的价值日益凸显。党的十九届四中全会从国家治理体系和治理能力现代化的高度将…

详解Qt绘图机制

Qt框架以其强大的图形界面功能著称,其中绘图机制是构建丰富视觉效果的关键。本文将详细介绍Qt中的绘图机制,包括绘图基础、绘图设备、绘图工具及高级特性,并通过实战C代码示例,带你领略Qt绘图的魅力。 绘图基础 Qt的绘图操作主要…

vs2019 - release版中_DEBUG宏生效的问题

文章目录 vs2019 - release版中_DEBUG宏生效的问题概述笔记总结END vs2019 - release版中_DEBUG宏生效的问题 概述 在加固程序,需要去掉PE的字符串表中和逻辑相关的字符串。 编译成release版后,用IDA看,还是发现有debug版才有的字符串。 那…

gitee关联picgo设置自己的typora_图床

一:去gitee官网创建仓库:typora_图床 1.百度搜索关键字:gitee,进入官网 2.进入gitee登录或者注册自己的账号 3.进入主页后,点击右上方 4.点击新建仓库 5.设置仓库名:typora_图床 6.点击5的创建&#xff0…

<计算机网络自顶向下> Internet Protocol(未完成)

互联网中的网络层 IP数据报格式 ver: 四个比特的版本号(IPV4 0100, IPV6 0110) headlen:head的长度(头部长度字段(IHL)指定了头部的长度,以32位字(4字节)为单位计算。这…

echarts实现水滴图

使用echarts实现水滴图 引入依赖&#xff0c;echarts-liquidfill3兼容echarts5; 安装依赖 "echarts": "^5.4.3","echarts-liquidfill": "^3.1.0",npm install echarts-liquidfill3.1.0 -S实现的效果图 构建一个水滴图的页面 <tem…

基于Spring Boot的商务安全邮件收发系统设计与实现

基于Spring Boot的商务安全邮件收发系统设计与实现 开发语言&#xff1a;Java框架&#xff1a;springbootJDK版本&#xff1a;JDK1.8数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/idea 系统部分展示 已发送效果图&#xff0c;用户可以对已发送信息…

4.28总结

对部分代码进行了修改&#xff0c;将一些代码封装成方法&#xff0c;实现了头像功能&#xff0c;创建一个本地字节输入流 fileinputSteam 对象&#xff0c;构造方法中绑定读取的数据源&#xff0c;创建一个socket对象&#xff0c;构造方法中绑定服务器的IP地址和端口号使用sock…

Kafka(十二)Streams

目录 Streams1 什么式是流式处理2 流式处理的相关概念2.1 拓扑2.2 时间2.2.1 输入时间2.2.2 输出时间 2.3 状态2.4 流和表2.5 时间窗口2.5.1 测试时间窗口 2.6 处理保证 3 流式处理设计模式3.1 单事件处理3.2 使用本地状态3.3 多阶段处理和重分区3.4 使用外部查找&#xff1a;流…

阿里云操作日记

昨天买了一个超级便宜的阿里云服务器&#xff0c;2核2G&#xff0c;3M固定带宽&#xff0c;40G ESSD Entry云盘&#xff0c;搭载一个简单的系统&#xff0c;就想到了docker轻量级&#xff0c;易于管理 其实docker很好用&#xff0c;第一步就是安装docker 一、docker安装与端口…

Python快速入门1数据类型(需要具有编程基础)

数据类型&#xff1a; Python 3.0版本中常见的数据类型有六种&#xff1a; 不可变数据类型可变数据类型Number&#xff08;数字&#xff09;List&#xff08;列表&#xff09;String&#xff08;字符串&#xff09;Dictionary&#xff08;字典&#xff09;Tuple&#xff08;元…

python学习笔记----循环语句(四)

一、while循环 为什么学习循环语句 循环在程序中同判断一样&#xff0c;也是广泛存在的&#xff0c;是非常多功能实现的基础&#xff1a; 1.1 while循环语法 while 条件表达式:# 循环体# 执行代码这里&#xff0c;“条件表达式”是每次循环开始前都会评估的表达式。如果条件…

张大哥笔记:我付钱了,我就是大爷?

很抱歉用这个当做标题&#xff0c;来给大家分享一些电商的故事&#xff01;大家好&#xff0c;我是张大哥&#xff0c;今天聊聊在电商路上遇到过的奇葩买家&#xff1f; 比如最近我在做PDD的时候&#xff0c;就会遇到很多莫名其妙的sha子&#xff0c;咱是知识份子&#xff0c;肯…
最新文章