在并发编程中有三个非常重要的特性:原子性、有序性,、可见性,学妹发现你对它们不是很了解,她很着急,因为理解这三个特性对于能够正确地开发高并发程序有很大的帮助,接下来的面试中也极有可能被问到,小学妹就忍不住开始跟你逐一介绍起来。

Java内存模型

在讲三大特性之前先简单介绍一下Java内存模型(Java Memory Model,简称JMM),了解了Java内存模型以后,可以更好地理解三大特性。

Java内存模型是一种抽象的概念,并不是真实存在的,它描述的是一组规范或者规定。JVM运行程序的实体是线程,每一个线程都有自己私有的工作内存。Java内存模型中规定了所有变量都存储在主内存中,主内存是一块共享内存区域,所有线程都可以访问。但是线程对变量的读取赋值等操作必须在自己的工作内存中进行,在操作之前先把变量从主内存中复制到自己的工作内存中,然后对变量进行操作,操作完成后再把变量写回主内存。线程不能直接操作主内存中的变量,线程的工作内存中存放的是主内存中变量的副本。

欢迎关注微信公众号: 万猫学社 ,每周一分享Java技术干货。

原子性(Atomicity)

什么是原子性

原子性是指:在一次或者多次操作时,要么所有操作都被执行,要么所有操作都不执行。

一般说到原子性都会以银行转账作为例子,比如张三向李四转账100块钱,这包含了两个原子操作:在张三的账户上减少100块钱;在李四的账户上增加100块钱。这两个操作必须保证原子性的要求,要么都执行成功,要么都执行失败。不能出现张三的账户减少100块钱而李四的账户没增加100块钱,也不能出现张三的账户没减少100块钱而李四的账户却增加100块钱。

原子性示例

示例一

i = 1;

根据上面介绍的Java内存模型,线程先把 i=1 写入工作内存中,然后再把它写入主内存,就此赋值语句可以说是具有原子性。

示例二

i = j;

这个赋值操作实际上包含两个步骤:线程从主内存中读取j的值,然后把它存入当前线程的工作内存中;线程把工作内存中的i改为j的值,然后把i的值写入主内存中。虽然这两个步骤都是原子性的操作,但是合在一起就不是原子性的操作。

示例三

i++;

这个自增操作实际上包含三个步骤:线程从主内存中读取i的值,然后把它存入当前线程的工作内存中;线程把工作内存中的i执行加1操作;线程再把i的值写入主内存中。和上一个示例一样,虽然这三个步骤都是原子性的操作,但是合在一起就不是原子性的操作。

从上面三个示例中,我们可以发现:简单的读取和赋值操作是原子性的,但把一个变量赋值给另一个变量就不是原子性的了;多个原子性的操作放在一起也不是原子性的。

如何保证原子性

在Java内存模型中,只保证了基本读取和赋值的原子性操作。如果想保证多个操作的原子性,需要使用 synchronized 关键字或者 Lock 相关的工具类。如果想要使int、long等类型的自增操作具有原子性,可以用java.util.concurrent.atomic包下的工具类,如: AtomicInteger 、 AtomicLong 等。另外需要注意的是, volatile 关键字不具有保证原子性的语义。

欢迎关注微信公众号: 万猫学社 ,每周一分享Java技术干货。

可见性(Visibility)

什么是可见性

可见性是指:当一个线程对共享变量进行修改后,另外一个线程可以立即看到该变量修改后的最新值。

可见性示例

package onemore.study;import java.text.SimpleDateFormat;import java.util.Date;public class VisibilityTest {    public static int count = 0;    public static void main(String[] args) {        final SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss.SSS");        //读取count值的线程        new Thread(() -> {            System.out.println("开始读取count...");            int i = count;//存放count的更新前的值            while (count < 3) {                if (count != i) {//当count的值发生改变时,打印count被更新                    System.out.println(sdf.format(new Date()) + " count被更新为" + count);                    i = count;//存放count的更新前的值                }            }        }).start();        //更新count值的线程        new Thread(() -> {            for (int i = 1; i <= 3; i++) {                //每隔1秒为count赋值一次新的值                try {                    Thread.sleep(1000);                } catch (InterruptedException e) {                    e.printStackTrace();                }                System.out.println(sdf.format(new Date()) + " 赋值count为" + i);                count = i;            }        }).start();    }}

在运行代码之前,先想一下运行的输出是什么样子的?在更新count值的线程中,每一次更新count以后,在读取count值的线程中都会有一次输出嘛?让我们来看一下运行输出是什么:

开始读取count...17:21:54.796 赋值count为117:21:55.798 赋值count为217:21:56.799 赋值count为3

从运行的输出看出,读取count值的线程一直没有读取到count的最新值,这是为什么呢?因为在读取count值的线程中,第一次读取count值时,从主内存中读取count的值后写入到自己的工作内存中,再从工作内存中读取,之后的读取的count值都是从自己的工作内存中读取,并没有发现更新count值的线程对count值的修改。

如何保证可见性

在Java中可以用以下3种方式保证可见性。

使用 volatile 关键字

当一个变量被 volatile 关键字修饰时,其他线程对该变量进行了修改后,会导致当前线程在工作内存中的变量副本失效,必须从主内存中再次获取,当前线程修改工作内存中的变量后,同时也会立刻将其修改刷新到主内存中。

使用 synchronized 关键字

synchronized 关键字能够保证同一时刻只有一个线程获得锁,然后执行同步方法或者代码块,并且确保在锁释放之前,会把变量的修改刷新到主内存中。

使用 Lock 相关的工具类

Lock 相关的工具类的 lock 方法能够保证同一时刻只有一个线程获得锁,然后执行同步代码块,并且确保执行 Lock 相关的工具类的 unlock 方法在之前,会把变量的修改刷新到主内存中。

欢迎关注微信公众号: 万猫学社 ,每周一分享Java技术干货。

有序性(Ordering)

什么是有序性

有序性指的是:程序执行的顺序按照代码的先后顺序执行。

在Java中,为了提高程序的运行效率,可能在编译期和运行期会对代码指令进行一定的优化,不会百分之百的保证代码的执行顺序严格按照编写代码中的顺序执行,但也不是随意进行重排序,它会保证程序的最终运算结果是编码时所期望的。这种情况被称之为 指令重排 (Instruction Reordering)。

有序性示例

package onemore.study;public class Singleton {    private Singleton (){}    private static boolean isInit = false;    private static Singleton instance;    public static Singleton getInstance() {        if (!isInit) {//判断是否初始化过            instance = new Singleton();//初始化            isInit = true;//初始化标识赋值为true        }        return instance;    }}

这是一个有问题的单例模式示例,假如在编译期或运行期时指令重排,把 isInit = true; 重新排序到 instance = new Singleton(); 的前面。在单线程运行时,程序重排后的执行结果和代码顺序执行的结果是完全一样的,但是多个线程一起执行时就极有可能出现问题。比如,一个线程先判断 isInit 为false进行初始化,本应在初始化后再把 isInit 赋值为true,但是因为指令重排没后初始化就把 isInit 赋值为true,恰好此时另外一个线程在判断是否初始化过, isInit 为true就执行返回了 instance ,这是一个没有初始化的 instance ,肯定造成不可预知的错误。

如何保证有序性

这里就要提到Java内存模型的一个叫做先行发生(Happens-Before)的原则了。如果两个操作的执行顺序无法从Happens-Before原则推导出来,那么可以对它们进行随意的重排序处理了。Happens-Before原则有哪些呢?

  • 程序次序原则:一段代码在单线程中执行的结果是有序的。
  • 锁定原则:一个锁处于被锁定状态,那么必须先执行 unlock 操作后面才能进行 lock 操作。
  • volatile变量原则:同时对 volatile 变量进行读写操作,写操作一定先于读操作。
  • 线程启动原则: Thread 对象的 start 方法先于此线程的每一个动作。
  • 线程终结原则:线程中的所有操作都先于对此线程的终止检测。
  • 线程中断原则:对线程 interrupt 方法的调用先于被中断线程的代码检测到中断事件的发生。
  • 对象终结原则:一个对象的初始化完成先于它的 finalize 方法的开始。
  • 传递原则:操作A先于操作B,操作B先于操作C,那么操作A一定先于操作C。

除了Happens-Before原则提供的天然有序性,我们还可以用以下几种方式保证有序性:

volatilesynchronizedLock

总结

  • 原子性:在一次或者多次操作时,要么所有操作都被执行,要么所有操作都不执行。
  • 可见性:当一个线程对共享变量进行修改后,另外一个线程可以立即看到该变量修改后的最新值。
  • 有序性:程序执行的顺序按照代码的先后顺序执行。

synchronized 关键字和 Lock 相关的工具类可以保证原子性、可见性和有序性, volatile 关键字可以保证可见性和有序性,不能保证原子性。

原文链接:https://www.tuicool.com/articles/nmiqEjY

java 原子类_小学妹教你并发编程的三大特性:原子性、可见性、有序性相关推荐

  1. 原子自增_小学妹教你并发编程的三大特性:原子性、可见性、有序性

    在并发编程中有三个非常重要的特性:原子性.有序性,.可见性,学妹发现你对它们不是很了解,她很着急,因为理解这三个特性对于能够正确地开发高并发程序有很大的帮助,接下来的面试中也极有可能被问到,小学妹就忍 ...

  2. 学妹教你并发编程的三大特性:原子性、可见性、有序性

    在并发编程中有三个非常重要的特性:原子性.有序性,.可见性,学妹发现你对它们不是很了解,她很着急,因为理解这三个特性对于能够正确地开发高并发程序有很大的帮助,接下来的面试中也极有可能被问到,小学妹就忍 ...

  3. 【Java 并发编程】 03 万恶的 Bug 起源—并发编程的三大特性

    今天让我们一起走进并发编程中万恶的Bug起源-并发编程中三大特性.今天学习目标如下: 并发编程的三大特性都要哪些 ? 并发编程三大特性的由来? 如何解决并发编程三大特性问题? 基本概念 原子性:一组操 ...

  4. 【Java】保证并发安全的三大特性

    一.并发编程三大特性的定义和由来 并发编程这三大特性就是为了在多个线程交替执行任务的过程中保证线程安全性. 二.为什么会出现线程不安全的现象呢? 接下来我们从这三个特性切入来介绍线程不安全的原因. 1 ...

  5. Java原子类中CAS的底层实现,java高级面试笔试题

    我总结出了很多互联网公司的面试题及答案,并整理成了文档,以及各种学习的进阶学习资料,免费分享给大家. 扫描二维码或搜索下图红色VX号,加VX好友,拉你进[程序员面试学习交流群]免费领取.也欢迎各位一起 ...

  6. 原子性 可见性 有序性_极简主义的内容可见性

    原子性 可见性 有序性 A couple of years ago, Minimalism as a concept took over the design world. 几年前, 极简主义作为一种 ...

  7. java内存模型 原子性_Java内存模型JMM 高并发原子性可见性有序性简介 多线程中篇(十)...

    JVM运行时内存结构回顾 在JVM相关的介绍中,有说到JAVA运行时的内存结构,简单回顾下 整体结构如下图所示,大致分为五大块 而对于方法区中的数据,是属于所有线程共享的数据结构 而对于虚拟机栈中数据 ...

  8. java 如何知道对象是否被修改过_Java 并发编程:AQS 的原子性如何保证

    当我们研究AQS框架时(对于AQS不太熟知可以先阅读<什么是JDK内置并发框架AQS>,会发现AbstractQueuedSynchronizer这个类很多地方都使用了CAS操作.在并发实 ...

  9. 【Java】《Java面向对象编程的三大特性》阅读笔记

    前言 偶然读到这篇文章(<Java面向对象编程的三大特性>),想来这也算论文?这种还不满网络都是?读罢觉得写得还真不错,这里以我愚见,简单点评一二,不足之处还望指教. 阅读笔记 笔记1 文 ...

最新文章

  1. h5 移动端 关于监测切换程序到后台或息屏事件和visibilitychange的使用
  2. JAVA条件表达式的陷阱
  3. android GLSurfaceView渲染模式
  4. tcp的无延时发送_腾讯网红程序员,详解带宽、延时、吞吐率、PPS 这些都是啥?...
  5. 湖人豪华助教团散伙 德帅身边恐仅剩两名帮手
  6. 如何利用WGET覆写已存在的档案
  7. ip变更会影响账号登陆吗_【教程】PUBG账号被盗导致封禁申诉解封教程
  8. MySQL笔记-左连接的使用(left join有关联的多表查询)
  9. Android 系统(220)---如何快速对系统重启问题进行归类
  10. UVA - 1267 Network
  11. PowerDesigner实用技巧小结(4)
  12. mysql 排查问题一些小技巧
  13. 计算机教案 认识键盘,《认识电脑键盘》教案
  14. 塔菲克蓝牙适配器驱动_TAFIQ蓝牙适配器4.0驱动下载
  15. matlab小波去噪wden,MATLAB小波去噪
  16. 小说阅读,原生小说APP源码出售,可二次开发 四端互通:android端,ios端,h5端,公众号端
  17. 交互设计理论之格式塔理论与四大法则
  18. CF1428F-Fruit Sequences
  19. 什么是产生式知识表示?给出这种表示方法的有缺点?
  20. 美国印钱 为什么不会通货膨胀

热门文章

  1. 【乡音】海安话四级考试
  2. 配置nginx-rtmp流媒体服务器(宝塔面板配置教程)
  3. java接口类支持多继承
  4. 数据结构与算法--1.整型变量值互换
  5. android怎样封装,如何封装属于自己的博客网站安卓APP 源码家园
  6. 深度学习之卷积神经网络(Convolutional Neural Networks, CNN)(二)
  7. mysql 之 优化 (收集于网络)
  8. 【Python】最新Python学习路线(完整详细版,含整套教程)
  9. Django 时间与时区设置问题
  10. Centos7:mysql5.6安装,配置及使用(RPM方式)