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

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

24395925ad6d4ad6161add7570853f89.png

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

关于a标签不能调用js方法的小细节,你注意到了么?

在我们做后台删除的时候&#xff0c;当点击删除标签时&#xff0c;你希望弹出一个友好的提示框&#xff01;比如这样&#xff1a; 那代码应该怎样写呢&#xff1f;向下面这样&#xff1f; <!DOCTYPE html> <html lang"en"> <head><meta charset…

android power 按键,Android Framework层Power键关机流程(一,Power长按键操作处理)

一&#xff1a;Android处理Power按键长按操作在Framework层中&#xff0c;Android4.x对Power键(KeyEvent.KEYCODE_POWER)的操作&#xff0c;我们从PhoneWindowManager开始分析&#xff0c;在分析前我这里打印了该方法的堆栈调用信息。大家可以参考一下。public long interceptK…

4岁小女孩给Linux内核贡献提交

今天在reddit上看到一个有趣的讨论&#xff0c;一个4岁的小女孩给Linux提交了一个补丁&#xff0c;并且这个补丁合并到了代码中。链接如下&#xff1a;https://www.reddit.com/r/linux/comments/2pqqla/kernel_commit_4_year_old_girl_fixes_formatting_to/cmzfvpl/我们看看她修…

【娱乐】收录各种神奇知乎问答

1、有哪些算法或数据结构是ACM大牛们在比赛中创造出来的&#xff1f; 2、c有哪些像__gcd这样的编译器自带函数&#xff1f; 3、为什么尽量不要使用using namespace std&#xff1f; 4、经常用 LaTeX 的是些什么人&#xff1f; 转载于:https://www.cnblogs.com/zjp-shadow/p/722…

php文件上传后没有打开权限_记墨者靶机文件上传(二)

“ 声明&#xff1a;该公众号大部分文章来自日常学习笔记&#xff0c;若是转载会先得到原作者授权或其他公众号白名单&#xff0c;并附上链接。剑者&#xff0c;心之刃也。既可为杀&#xff0c;亦可为护。杀与护&#xff0c;不过一念之间&#xff01;请勿利用文章内的相关技术从…

你还会写这段C51程序吗?

经典题目解析定时器T1采用计数模式&#xff0c;方式1中断&#xff0c;计数输入引脚 P3.5外接开关按钮作为计数信号输入&#xff0c;P1口控制8个LED小灯&#xff0c;初始状态所有小灯全亮&#xff0c;按3次按钮开关产生计数中断时&#xff0c;高3位和低5位交替闪烁3次&#xff0…

鸿蒙系统能不能用了,【图片】华为鸿蒙系统的厉害之处在于 你可能非用不可 !【手机吧】_百度贴吧...

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼鸿蒙的厉害在于 你可能非用不可&#xfffc;瀑布先生06-05 09:52科技达人关注&#xfffc;华为一直在研发一套操作系统&#xff0c;如果不是对手打压。可能大部分人&#xff0c;至今都不知道这个系统的存在。如果市场稳定&#xff…

JDBC连接数据库(Java DataBase Connectivity,java)

通过JDBC操作数据库(以mysql为例) 1、官网下载JDBC的相关JAR包 【https://dev.mysql.com/downloads/connector/j/】 2、解压后&#xff0c;导入jar包 连接数据(引入外部jar包)项目上右键->build path->configure build path->上面Libraries->Add External JARs 3、…

python刘卫国实验题_MATLAB 程序设计与应用(刘卫国版)习题答案3-4

习题3&#xff1a;1.>> syms x f>>flimit((cos(sqrt(x)))^(pi/x),x,0,right)f exp(-1/2*pi)2.>> syms x f>>flimit((3*sin(x)x^2*cos(1/x))/((1cos(x))*log(1x)),x,0)f 3/23.>> syms x f>>flimit((sqrt(4*x^2x-1)x1)/sqrt(x^2sin(x)),x,-…

过年回家抢票攻略

每年过年抢票都是一个困难的事情&#xff0c;所以决定发一篇文章汇总一下各种抢票软件&#xff0c;大家自己有方案的可以在文章后面留言&#xff0c;让跟多的人看到&#xff0c;希望大家都能顺利抢到车票&#xff0c;顺利回家。超级抢票机&#xff0c;一款超强的火车票抢票神器…

[小技巧]diff的文件夹忽略使用方式

当我们比较两个文件夹时经常需要忽略.svn或者.git&#xff0c;那么如下 diff -r -x ".git" -x "*.ko" -x "*.o" -x "cscope*" -x "*.map" path1 path2 转载于:https://www.cnblogs.com/aaronLinux/p/7232505.html

python空列表添加_Python列表的简单操作

Python列表的简单操作&#xff1a;1. 在列表末尾添加元素&#xff1a;sth.append()2. 在列表中添加元素&#xff1a;sth.insert(位置&#xff0c;元素)3. 从列表中删除元素&#xff1a;1) del sth[元素位置] 2) pop的元素名 sth.pop(元素位置&#xff0c;不填则默认列表末尾)&…

C语言,函数不可返回指向栈内存的指针

预备知识&#xff1a;内存的分类C/C程序占用的内存分为两大类&#xff1a;静态存储区与动态存储区。其示意图如下所示&#xff1a;数据保存在静态存储区与动态存储区的区别就是&#xff1a;静态存储区在编译-链接阶段已经确定了&#xff0c;程序运行过程中不会变化&#xff0c;…

android 界面绘制完毕,几种获取android 界面性能数据的快捷方法

探测 界面绘制性能获取界面的绘制性能有很多种方法&#xff0c;比如说 Systrace 但是这种方法 有一个不太好的地方就是使用起来较为复杂&#xff0c; 有没有一种 谷歌官方推荐 的方便一点的方法 &#xff0c;其实是有的&#xff0c;只需要一个函数 就可以获得layout的时间 非常…

联想rd650怎么装系统win7_Lenovo g50重装win7系统|U盘重装联想g50笔记本系统

Lenovo可以说是个家喻户晓的品牌很多小伙伴们也是使用联想的电脑&#xff0c;今天有小伙伴和小编提了一款Lenovo g50笔记本&#xff0c;小编了解到这款笔记本电脑的口碑和销量都很不错&#xff0c;其实这位小伙伴的问题呢是要如何重装win7系统&#xff0c;毕竟质量再好的电脑也…

书籍推荐

今天天气很好&#xff0c;风和日丽&#xff0c;艳阳高照&#xff0c;大家心情应该也很不错&#xff0c;毕竟&#xff0c;今天是周五&#xff0c;就像上学的时候一样&#xff0c;下午的铃声一响&#xff0c;每个同学都像脱缰的野马一样&#xff0c;周五&#xff0c;上班族的人们…

【转】JMeter学习(十三)分布式部署

Jmeter 是Java 应用&#xff0c;对于CPU和内存的消耗比较大&#xff0c;因此&#xff0c;当需要模拟数以千计的并发用户时&#xff0c;使用单台机器模拟所有的并发用户就有些力不从心&#xff0c;甚至会引起JAVA内存溢出错误。为了让jmeter工具提供更大的负载能力&#xff0c;j…

android mvp框架基类,Android MVP架构模式基类封装

前言MVP模式是Android官方推荐的架构模式&#xff0c;可使视图与数据层完全解耦。本文旨意封装在MVP模式中的基类如Activity&#xff0c;Fragment&#xff0c;Presenter类。以下内容建议在了解了mvp模式的读者阅读&#xff0c;如果还有对mvp架构模式有疑问的&#xff0c;请看我…

C语言,字符串指针做函数参数

看一下下面这段代码有什么问题&#xff1f;#include "stdio.h" //#include "stdbool.h" #include "string.h" #include "stdlib.h" #include "math.h"void getMemory(char *p) {/*char *p str*/p (char *)malloc(100);str…

java创建一个未知长度的数组_Java数组的创建操作

数组是一个固定长度的&#xff0c;包含了相同类型数据的 容器步骤1:声明数组步骤2:创建数组步骤3:访问数组步骤4:数组长度步骤5:练习-数组最小值步骤6:答案-数组最小值步骤 1 : 声明数组int[] a; 声明了一个数组变量。[]表示该变量是一个数组int 表示数组里的每一个元素都是一个…