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知识库 -> DCL单例到底需不需要volatile? -> 正文阅读

[Java知识库]DCL单例到底需不需要volatile?

前言知识介绍

首先我们来看下面一道程序,这道程序很简单,请问这个程序的运行结果是什么?8还是0?

public class Escape {
    private int num = 8;

    private Escape() {
            new Thread(() -> System.out.println(this.num)).start();
    }

    public static void main(String[] args) throws IOException {
        new Escape();
        System.in.read();
    }
}

对象的创建过程

java程序运行过程中每时每刻读有对象被创建出来,在语言层面上,创建对象仅仅是一个new关键字而已,而在字节码层面中对象的创建过程又是怎样的呢?

 LINENUMBER 21 L0
    NEW lixinzhen/Escape
    DUP
    INVOKESPECIAL lixinzhen/Escape.<init> ()V
    POP

这是对前面程序反编译的字节码,从中可以看出对象的创建需要经历三个步骤,第一步申请内存,给对象赋予默认值0(此时成为半初始化状态),第二步调用构造方法初始化这个对象赋值为8,第三步建立连接(执行到INVOKESPECIAL这里对象已经创建完毕)。(DUP这个指令涉及到堆栈的调用,过于复杂这里笔者不予介绍)

半初始化

在对象的创建过程中笔者提到半初始状态,有没有想过为什么要有半初始化状态也就是说为什么要给num对象赋予初始值为0呢?原因在于给num对象分配的内存区域可能之前存在某个值,而为了防止num的值被错误分配,于是先给定一个初始值。(这也是为什么Java比C/C++更加安全的原因,早期的c/c++指针是暴露在外面的,可以随时窃取到密码)

DCL(Double Check Lock)

在设计模式中最简单的就是单例模式,那么什么是单例模式呢?笔者相信大家都知道,所谓单例模式就是在整个类的运行过程中,“有且仅有”一个对象实例。笔者曾经对单例模式也进行过相关介绍,如果不太了解可以先看看这篇文章。饿汉模式和懒汉模式

在看完这篇文章后,你可以发现在单例模式中保证线程安全的最优方案应该就是DCL,在这里笔者将其代码复制下来以便后续分析。

public class SingletonLazy {
    private static SingletonLazy singleton;
    public SingletonLazy() {
    }
    public static SingletonLazy getInstance() {
        //再次判断当前singleton是否已经创建,如果创建,直接获取
        if (singleton == null) {
            synchronized (SingletonLazy.class) {
                //先判断当前对象是否已经创建
                if (singleton == null) {
                    singleton = new SingletonLazy();
                }
            }
        }
        return singleton;
    }
}
public class SingletonTest {
    public static void main(String[] args) {
        SingletonLazy instance = SingletonLazy.getInstance();
        SingletonLazy instance2 = SingletonLazy.getInstance();
        System.out.println(instance==instance2);//true
    }
}

在这里笔者想提一个问题,最上面的那个if语句能不能去掉?

我们知道加锁是一个非常耗资源耗性能的事情,如果说加锁需要100ns,那么if语句只需要1ns。如果有1W个线程同时进来,此时去掉了最上面的if语句的话,那么就需要对这个1W个线程进行校验也就是说需要对这1W个线程全部加锁,那么造成的性能损耗是非常严重的。而加了if语句后,可以方便的对这个1W个线程进行判断。

volatile关键字

Java并发编程有3大特性,即可见性,有序性,原子性。如果你看过《深入理解Java虚拟机》这本书你就应该知道在Java内存模型中始终存在乱序问题,因此可见性和原子性可以保证,但有序性却很难保证。

voltalie关键字大家应该比较熟悉,它有两方面的作用:保证可见性和禁止重排序。

那么它是如何做到的呢?其实归根到底就是Lock 指令。(这是汇编指令,这里不细讲,后续笔者会对缓存一致性协议进行相关文章编写的时候会讲到)

指令重排序

指令重排序(处理器的乱序执行)是指源码顺序和程序顺序不一样,指令重排序不是必然发生的,指令重排序会导致线程安全问题。(也就是说单线程结果的最终一致性没有问题,可以随便换)

在单线程环境下指令重排序没有影响,但在多线程环境下指令重排序很有可能会乱序执行。前面讲到的volatile关键字可以避免指令重排序。

最后

人生有三大境界:看山是山,看山不是山,看山还是山。笔者能力有限,只能写出这么多,如有不当之处还请多指教。

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-02-28 15:14:49  更:2022-02-28 15:17:11 
 
开发: 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 11:26:43-

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