【JUC】十八、happens-before先行发生原则

文章目录

  • 1、先行发生原则happens-before
  • 2、happens-before总原则
  • 3、8条happens-before规则
  • 4、案例

1、先行发生原则happens-before

在Java中,Happends-Before本质上是规定了一种可见性, A Happends-Before B,则A发生过的事情对B来说是可见的,不论A事件和B事件是否发生在同一个线程里。

  • happens-before体现的是对可见性和有序性的约束。
  • happens-before是并发环境下,两个操作是否可能存在冲突的判断依据

在这里插入图片描述

y一定等于5

如果线程A的操作(x=5)先行发生于线程B的操作(y=x),或者说这两个事件存在先行发生原则,那y=5一定成立,反之则不一定,因为x=5的改变可能还没从A线程的工作内存刷回主内存,线程就暂时挂起了。

但如果Java内存模型中的有序性都得靠volatile和synchronized来实现,就非常繁琐,而且日常开发也没见处处加这些关键字,这是因为谁先谁后在先行发生原则里已经立好了规矩。

2、happens-before总原则

1)如果一个操作happens-before另一个操作,那么第一个操作的执行结果将对第二个操作可见,而且第一个操作的执行顺序排在第二个操作之前。

2) 两个操作之间存在happens-before关系,并不意味着一定要按照happens-before原则制定的顺序来执行,如果重排序之后的执行结果与按照happens-before关系来执行的结果一致,那么这种重排序并不非法

举例:

值日表里写了周一张三,周二李四,但现在张三周一临时有事,和李四换班后,教室还是能打扫干净

3、8条happens-before规则

1> 次序规则一个线程内,按照代码顺序,写在前面的操作先于写在后面的操作

直白讲就是:同一个线程,前面一个操作把变量x赋值为1,那后面一个操作肯定知道x已经变成1了

2> 锁定规则:一个unLock操作先行发生于后面(这里的"后面"是指时间上的先后)对同一个锁的lock操作

直白说就是:一定是:A线程unlock后,B线程才能对同一个锁lock

Lock lock = new ReentrantLock();lock.lock();
try{}finally{//先lock.unlock();
}
//后
lock.lock();
try{}finally{lock.unlock();
}

3> volatile变量规则 :对一个volatile变量的写操作先行发生于后面对这个变量的读操作,前面的写对后面的读是可见的,这里的”后面“同样是指时间上的先后

4> 传递规则:如果操作A先行发生于操作B,而操作B又先行发生于操作C,则可以得出操作A先行发生于操作C

5> 线程启动规则:Thread对象的start()方法先行发生于此线程的每一个动作

执行顺序一定是第4行先,第2行后:

Thread t1 = new Thread(() -> {System.out.println("QWE");   //后},"t1");
t1.start();  //先

6> 线程中断规则:对线程interrupt()方法的调用先行发生于被中断线程的代码检测到中断事件的发生,可以通过Thread.interrupted()检测到是否发生中断。

直白说:检测到中断事件发生(检测到中断标志位变了),是先发生了interrupt方法的调用。一定是先发烧了,温度计才能检测到体温变了。

7> 线程终止规则:线程中的所有操作都先行发生于对此线程的终止检测,我们可以通过isAlive()等手段检测线程是否已经终止执行

直白说:线程中的操作(比如run方法体)先全部执行完,线程才终止

8> 对象终结规则:一个对象的初始化完成(构造函数执行结束) 先行发生于它的finalize()方法的开始

翻译:肯定是先new了一个对象,才能垃圾回收这个对象

4、案例

private int value = 0;private int getValue(){return value;
}private int setValue(){return ++value;
}

现在有线程A和线程B,线程A(在时间上先)调用了setValue方法,然后线程B调用同一对象的getValue方法,那线程B的返回值是?

对照上面的8条规则:
  • 两个方法在不同线程,第一条规则用不上
  • 两个方法都未加锁,规则2也pass
  • 共享变量value没有加volatile,规则3pass
  • 先行发生规则不等价于时间上的先,这里目前直接没有已知的先行发生规则,无从传递,pass

⇒ 无法通过happens-before原则推导出线程A happens-before线程B,虽然可以确认在时间上线程A优先于线程B,无法确认线程B获得的结果是什么,所以这段代码不是线程安全的。

怎么修复?

方式一:加synchronized,如下,这样性能损失太大

private int value = 0;private synchronized int getValue(){return value;
}private synchronized int setValue(){return ++value;
}

方式二:把value定义为volatile变量,由于setter方法对value的修改不依赖value的原值,满足volatile关键字使用场景

private volatile int value = 0;private int getValue(){return value;  //利用volatile保证读取操作的可见性
}private synchronized int setValue(){  //利用synchronized保证复合操作的原子性return ++value; 
}

利用volatile保证读取操作的可见性,利用synchronized保证复合操作的原子性,结合使用锁和volatile 变量来减少同步的开销

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

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

相关文章

Discuz论坛自动采集发布软件

随着网络时代的不断发展,Discuz论坛作为一个具有广泛用户基础的开源论坛系统,其采集全网文章的技术也日益受到关注。在这篇文章中,我们将专心分享通过输入关键词实现Discuz论坛的全网文章采集,同时探讨采集过程中伪原创的发布方法…

企业微信hook接口调用,批量消息id转发

简要描述 批量消息id转发 操作码 101029 请求示例 { "type": 101029, "data": { "sendId": [ "78813025xxx38" ], "msg_id": "1119934", "msg_type": 23, "syncKey": "1680696363&q…

【Appium】解决搜索输入框无搜索按钮

问题:被测组件为搜索输入框,但是无搜索按钮,需要点击键盘的回车按钮,来进行搜索 使用环境:appium 、 python、 Android手机 操作背景 如果使用appium 输入中文字符,则需要设置’unicodeKeyboard’: True…

.net-去重的几种情况

文章目录 前言1. int 类型的list 去重2. string类型的 list 去重3. T泛型 List去重4. 使用HashSet List去重5. 创建静态扩展方法 总结 前言 .net 去重的几种情况 1. int 类型的list 去重 // List<int> List<int> myList new List<int>(){ 100 , 200 ,100…

Selenium定位元素的方法css和xpath的区别!

selenium是一种自动化测试工具&#xff0c;它可以通过不同的定位方式来识别网页上的元素&#xff0c;如id、name、class、tag、link text、partial link text、css和xpath。 css和xpath是两种常用的定位方式&#xff0c;它们都可以通过元素的属性或者层级关系来定位元素&#…

Win10任务栏卡死?三个技巧,让你轻松应对!

windows 10作为广受欢迎的操作系统&#xff0c;为用户提供了强大的功能和友好的用户界面。然而&#xff0c;有时用户可能会面临任务栏卡死的问题&#xff0c;这不仅影响使用体验&#xff0c;还可能导致一系列其他问题。本文将深入介绍win10任务栏卡死的原因&#xff0c;并提供三…

【Linux】-信号-(信号的产生,保存,处理,以及os是怎么读取硬件的输入,硬件异常和coredump,定时器的原理简单的用户态和内核态的详细介绍)

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树&#x1f388; &#x1f389;作者宣言&#xff1a;认真写好每一篇博客&#x1f4a4; &#x1f38a;作者gitee:gitee✨ &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 动态规划算法&#x1f384; 如 果 你 …

AR增强现实在汉语文学课堂教学中的应用

一、AR增强现实在汉语文学课堂教学中的应用优势 1、增强互动性&#xff1a;AR技术可以将虚拟元素与现实环境相结合&#xff0c;使学生在学习过程中更加投入。例如&#xff0c;在讲解诗歌时&#xff0c;可以通过AR技术将诗歌中的景象生动地展示在学生面前&#xff0c;增强学生的…

外贸B2B网站独立站建站(零基础全流程)

1.第一步是要先去买个域名&#xff1a; 一般做外贸的购买.com 后缀的国际域名就好&#xff0c;域名可以在阿里云&#xff0c;腾讯云等大的平台上购买&#xff0c;方法很简单&#xff08;但是在确定购买新的域名最好要分析下这个域名有没有被黑过&#xff0c;要不然后期对这个网…

【OpenGL】Clion配置

OpenGL简介 OpenGL&#xff08;Open Graphics Library&#xff09;是指定义了一个跨编程语言、跨平台的编程接口规格的专业的图形程序接口。它用于三维图像&#xff08;二维的亦可&#xff09;&#xff0c;是一个功能强大&#xff0c;调用方便的底层图形库。OpenGL是行业领域中…

leetcode704. 二分查找

题目描述 给定一个 n 个元素有序的&#xff08;升序&#xff09;整型数组 nums 和一个目标值 target &#xff0c;写一个函数搜索 nums 中的 target&#xff0c;如果目标值存在返回下标&#xff0c;否则返回 -1。 示例 1: 输入: nums [-1,0,3,5,9,12], target 9 输出: 4 解…

01背包问题动态规划解法

二维数组dp #include<iostream> #include<vector> using namespace std;vector<int>weight { 1,3,4 }; vector<int>value { 15,20,30 }; int bagWeight 4; vector<vector<int>>dp(weight.size() 5, vector<int>(bagWeight 5, …

JVS低代码按钮组件触发逻辑,打破传统功能界限

在现代应用开发中&#xff0c;按钮组件的功能不仅仅局限于触发页面上的简单动作&#xff0c;它更可以成为连接前后端数据交互的桥梁。当按钮被点击时&#xff0c;其背后可能隐藏着复杂的逻辑远程调用过程&#xff0c;这些过程旨在从远程服务器获取数据&#xff0c;并将这些数据…

微信怎么设置自动回复

微信作为一款广受欢迎的社交媒体平台&#xff0c;其聊天功能是非常重要的。许多用户都希望能够快速、自动地回复消息 首先&#xff0c;点击设置&#xff0c;选择机器人下面的自动通过好友 点击新增规则&#xff0c;设置你自动通过好友的时间段&#xff0c;自动通过好友的微信工…

LeetCode Hot100 3.无重复字符的最长子串

题目&#xff1a; 给定一个字符串 s &#xff0c;请你找出其中不含有重复字符的 最长子串 的长度。 代码&#xff1a; class Solution {public int lengthOfLongestSubstring(String s) {char[] arr s.toCharArray(); // 转换成 char[] 加快效率&#xff08;忽略带来的空间…

文章解读与仿真程序复现思路——太阳能学报EI\CSCD\北大核心《基于电力CPS联合仿真的主配一体化电网负荷转供方法研究》

这个标题涉及到电力系统、CPS&#xff08;Cyber-Physical Systems&#xff0c;即网络化物理系统&#xff09;联合仿真&#xff0c;以及主配一体化电网负荷转供方法的研究。让我们逐步解读&#xff1a; 基于电力CPS联合仿真&#xff1a; 基于电力&#xff1a; 指的是该研究与电力…

python-nmap库使用教程(Nmap网络扫描器的Python接口)(功能:主机发现、端口扫描、操作系统识别等)

文章目录 Python-nmap库使用教程前置条件引入python-nmap创建Nmap扫描实例执行简单的主机发现&#xff08;nmap -sn&#xff09;示例&#xff0c;我有一台主机配置为不响应 ICMP 请求&#xff0c;但使用nmap -sn&#xff0c;仍然能够探测到设备&#xff1a; 端口扫描扫描特定端…

AIGC实战——生成对抗网络(Generative Adversarial Network)

AIGC实战——生成对抗网络 0. 前言1. 生成对抗网络1.1 生成对抗网络核心思想1.2 深度卷积生成对抗网络 2. 数据集分析3. 构建深度卷积生成对抗网络3.1 判别器3.2 生成器3.3 DCGAN 模型训练 4. GAN 训练技巧4.1 判别器强于生成器4.2 生成器强于判别器4.3 信息量不足4.4 超参数 小…

java 并行流parallelStream 与 CompletableFuture的选择

在 Java 中&#xff0c;并行流 (parallelStream) 和 CompletableFuture 都是处理并发和异步编程的工具&#xff0c;但它们在使用场景和适用性上有一些区别。下面是一些指导原则&#xff0c;可以帮助你在选择使用并行流或 CompletableFuture 时做出决策&#xff1a; 使用场景&a…

数据结构与算法之美学习笔记:29 | 堆的应用:如何快速获取到Top 10最热门的搜索关键词?

目录 前言堆的应用一&#xff1a;优先级队列堆的应用二&#xff1a;利用堆求 Top K堆的应用三&#xff1a;利用堆求中位数解答开篇内容小结 前言 本节课程思维导图&#xff1a; 搜索引擎的热门搜索排行榜功能你用过吗&#xff1f;搜索引擎每天会接收大量的用户搜索请求&#x…