java两个线程的通信/指令重排

目录

1.线程通信

2.指令重排

1.线程通信

前段时间面试笔试题,手写两个线程之间的通信。

问题回顾:

一个生产者,一个消费者,一个桌子媒介。两个线程分别是消费者和生产者。流程是生产者生产一个件商品,通知消费者消费一个商品。

普通的线程使用的等待和唤醒是使用Object类下的wait()notify()notifyAll()interrupt()四个方法。

wait()阻塞线程

notify()随机唤醒线程

notifyAll()唤醒全部线程

interrupt()打断线程。

如果要唤醒指定的线程需要使用Condition newCondition()的这个方法

实现demo

Desk.class

public class Desk {//上货标识public static boolean flag = false;//订单数量public static Integer orderCount = 0;//商品数量public static Integer productCount = 0;//最终剩余数量public static Integer finalCount =0;//定义可重入锁public static final Lock lock = new ReentrantLock();//给consumer使用的conditionpublic static Condition condition4Consumer = lock.newCondition();//给producer用的conditionpublic static Condition condition4Producer = lock.newCondition();
}

Producer.class

public class Producer extends Thread {@Overridepublic void run() {
//        int i=10;while (true) {Desk.lock.lock();try {if (!Desk.flag) {Thread.sleep(new Random().nextInt(10));Desk.flag = true;System.out.println(Thread.currentThread().getName() + ":用户生产食物" + (++Desk.productCount));Desk.finalCount++;System.out.println("商家唤醒consumer--");Desk.condition4Consumer.signalAll();} else {try {Desk.condition4Producer.await();} catch (InterruptedException e) {throw new RuntimeException(e);}}} catch (InterruptedException e) {throw new RuntimeException(e);} finally {Desk.lock.unlock();}}}
}

 Consumer.class

public class Consumer extends Thread {@Overridepublic void run() {
//        int i=10;while (true) {//不断地拿食物Desk.lock.lock();try {if (Desk.flag) {//如果有食物 拿走并通知厨师Thread.sleep(new Random().nextInt(10));System.out.println(Thread.currentThread().getName() + ":顾客拿走食物" + (++Desk.orderCount));Desk.finalCount--;Desk.flag = false;System.out.println("consumer 转杯唤醒producer--");Desk.condition4Producer.signalAll();} else {Desk.condition4Consumer.await();}} catch (InterruptedException e) {throw new RuntimeException(e);} finally {Desk.lock.unlock();}}}
}

测试类

public class TestCondition {public static void main(String[] args) throws InterruptedException {Producer producer = new Producer();Consumer consumer = new Consumer();producer.start();consumer.start();int numRandom = new Random().nextInt(200)*10;Thread.sleep(numRandom);producer.interrupt();consumer.interrupt();producer.join();consumer.join();System.out.println("等待时间:"+numRandom+"生产数量:"+Desk.productCount+"消费数量:"+Desk.orderCount+"最终数量:"+Desk.finalCount);producer.wait();}
}

最终结果

2.指令重排

指令重排的复现代码

在代码执行的过程中会进行必要的优化,印美每一次操作都需要消耗内存的操作

每一条指令的操作都是1.加载2.设置3.存储。那么对一个值的操作中会存在反复读取的操作,反复存储,那么需要把反复读取和存储的最终保障一次就可以。

在多线程中,现在要先修改ab的值,在把x=b.y=a,正常线程中cpu争抢线程顺序不一样,那么结果是1,1)(1,0)(0,1)。单如果是(0,0),此时就表示已经出现指令重排的现象。

public class TestReorder {static int x = 0;static int y = 0;static int a = 0;static int b = 0;static int count=0;public static void main(String[] args) throws InterruptedException {while (true) {CountDownLatch countDownLatch = new CountDownLatch(1);Thread t1 = new Thread(() -> {try {countDownLatch.await();} catch (InterruptedException e) {throw new RuntimeException(e);}//操作1a = 1;//操作2x = b;});Thread t2 = new Thread(() -> {try {countDownLatch.await();} catch (InterruptedException e) {throw new RuntimeException(e);}//操作1b = 1;//操作2y = a;});t1.start();t2.start();countDownLatch.countDown();t1.join();t2.join();//三种情况(1,1)(1,0)(0,1)//如果两个两个线程都发生重排序结果是(0,0)System.out.println("x" + x + "y" + y+"count"+count);if (x == 0 && y == 0) {break;}count++;x=0;y=0;a=0;b=0;}}
}

那么这个时候就需要使用使用volatile这个关键字。它是通过前后两个操作中添加屏障防止指令重排。

volatile通过使用内存屏障防止指令重排

写volatile操作前加storestore屏障(写写屏障)

写volatile操作后加storeload屏障(写读屏障)

读volatile操作后加loadload屏障(读读屏障)

读volatile操作后加loadstore屏障(读写屏障)

 ps:

volatile中主要又有两个作用,一个是可见性和防止指令重排。

可见性表示在多线程中,有一个变量修改,其他线程能立即察觉出它的变化中,而不是从缓存行中读取数据。

指令重排表示的是。正常是初始化是三步骤,第一开辟空间,第二初始化数据结构,第三指针指向堆空间。这这个的过车中其实就是,int a=1;的过程。但是从上到下在代码执行优化的时候会导致指令重排。

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

建立用邻接矩阵表示的无向图

建立用邻接矩阵表示的无向图 #include<stdio.h> #define NUM 100 typedef struct {char vexs[NUM];int edges[NUM][NUM];int n,e; }MGraph; void CreateMGraph(MGraph *G) {int i,j,k;printf("请输入顶点数和边数:");scanf("%d,%d",&G->n,&a…

大模型微调技术 --> LoRA 系列之 AdaLoRA

AdaLoRA 1.摘要 之前的微调方法(如低秩更新)通常将增量更新的预算均匀地分布在所有预训练的权重矩阵上&#xff0c;并且忽略了不同权重参数的不同重要性。结果&#xff0c;微调结果不是最优的。 为了弥补这一差距&#xff0c;我们提出了AdaLoRA&#xff0c;它根据权重矩阵的…

革新网络管理:拉线网套技术引领行业新趋势

根据QYResearch调研团队的最新力作《全球拉线网套市场报告2023-2029》预测&#xff0c;到2029年&#xff0c;全球拉线网套市场的规模有望达到11.7亿美元&#xff0c;且在未来几年内&#xff0c;将以3.5%的复合年增长率&#xff08;CAGR&#xff09;持续扩张。 在全球范围内&…

HTB:Devel[WriteUP]

目录 连接至HTB服务器并启动靶机 1.What is the name of the service is running on TCP port 21 on the target machine? 使用nmap对靶机TCP端口进行开放扫描 2.Which basic FTP command can be used to upload a single file onto the server? 尝试匿名连接至靶机FTP服…

Hadoop集群的高可用(HA)-(2、搭建resourcemanager的高可用)

第一步&#xff1a;检查mapred-site.xml &#xff0c;里面只有yarn配置和historyServer的配置&#xff0c;不需要修改 第二步&#xff1a;修改yarn-site.xml <?xml version"1.0"?> <!-- Licensed under the Apache License, Version 2.0 (the "Lic…

golang笔记

golang笔记 一、内存逃逸 本应在栈中内存,被分配到了堆中 1 返回指针对象 在外部被使用 2 reutrn 函数 使用了上面方法的敞亮 3 入参是interface{} 动态参数 4 make超过栈大小 -gcflags"-m"查看分配内存信息 返回变量vs返回指针 返回变量, 会多一步复制变量, 返回…

纹理分析——统计分析方法

一. 灰度共生矩阵法(Gray Level Co-occurrence Matrix, GLCM ) 灰度共生矩阵又称为灰度空间相关矩阵&#xff0c;是通过研究灰度的空间相关特性来描述纹理的常用方法。&#xff08;也称为联合概率矩阵&#xff09;它作为传统的图像纹理分析方法已广泛应用于数字图像处理的许多…

IT维修记录表导入接口的思路

上篇文章讲了IT设备信息表的导入接口的思路&#xff0c;这篇文章趁热打铁&#xff0c;把IT维修记录表的导入接口的思路给说一下。 首先我们要知道IT维修记录表的数据是什么来的&#xff1f;这个问题必须要搞懂&#xff0c;不搞懂的话对接下来的思路其实是不利的。IT维修记录表…

场景解决方案丨迎战电商大促,企业管理跟踪驾驶舱助力中小企业打赢决胜之战

该方案已沉淀为➡️订单物流信息跟踪模板&#xff0c;点击&#x1f517;即可体验 随着互联网技术的发展和市场经济的变化&#xff0c;各行业的线上竞争愈发激烈。一方面&#xff0c;互联网平台凭借便捷的服务和丰富的产品吸引了大量客户&#xff1b;另一方面&#xff0c;复杂多…

WebRTC 环境搭建

主题 本文主要描述webrtc开发过程中所需的环境搭建 环境&#xff1a; 运行环境&#xff1a;ubuntu 20.04 Node.js环境搭建 安装编译 Node.js 所需的依赖包: sudo apt-get update sudo apt-get install -y build-essential libssl-dev 下载 Node.js 源码: curl -sL htt…

Python从入门到高手7.5节-实现冒泡排序算法

目录 7.5.1 排序算法简介 7.5.2 冒泡排序算法原理 7.5.3 冒泡排序算法实现 7.5.4 永不放弃 7.5.1 排序算法简介 所谓排序&#xff0c;是指将数据集合中的元素按从小到大的顺序进行排列&#xff0c;或按从大到小的顺序进行排列。 前者称为升序排序&#xff0c;后者称为降序…

vue-quill-editor富文本编辑器

效果图&#xff1a; 1、下载安装vue-quill-editor npm install vue-quill-editor --save图片缩放、拖拽 npm install quill-image-drop-module -S //允许粘贴图像并将其拖放到编辑器中。 npm install quill-image-resize-module -S //允许调整图像大小<template>&…

TCP是怎样工作的网络拥塞控制理论和算法部分记录

参考资料 https://github.com/ituring/tcp-book 流量控制、窗口控制和拥塞控制的关系 流量控制、窗口控制和拥塞控制的关系如图所示 窗口控制是上层的概念&#xff0c;核心思路是基于滑动窗口技术传输数据。而确定发送窗口大小的方法有流量控制和拥塞控制两种 流量控制&…

NVR管理平台EasyNVR多个NVR同时管理对接天翼云云存储的一些关键信息和优势

在视频监控领域&#xff0c;随着技术的不断进步&#xff0c;存储方式的选择变得尤为重要。传统的本地存储方式受限于硬件容量&#xff0c;而云存储则以其强大的数据处理能力和弹性扩展性&#xff0c;成为视频数据存储的理想选择。NVR管理平台EasyNVR作为一款领先的视频汇聚与管…

饲料加工机器设备由搅拌机粉碎机颗粒机组成

饲料加工机器设备在现代养殖业中扮演着至关重要的角色&#xff0c;它们不仅提高了饲料的生产效率&#xff0c;还优化了饲料的营养价值。其中&#xff0c;饲料粉碎机、搅拌机和颗粒机是饲料加工流程中的三大核心设备。 想象一下&#xff0c;一把把粗糙的原料&#xff0c;在粉碎…

oracle数据坏块处理(二)-逻辑坏块重新格式化处理

1、问题描述 在使用duplicate搭建DG时报错 包括rman copy&#xff0c;rman备份 2、问题分析 由于数据文件逻辑坏块导致物理备份不能正常进行。 使用rman检查数据文件47 SELECT tablespace_name, segment_type, owner, segment_name FROM dba_extents WHERE file_id 47 a…

在IDEA使用arthas实现jar包方法耗时统计

1.背景 对于依赖jar包中的方法内部耗时统计&#xff0c;传统的手写StopWatch不适用&#xff0c;这儿采用arthas统计。 官网文档比较详细&#xff0c;trace | arthas 使用版本&#xff1a; arthas-boot version: 4.0.2 IntelliJ IDEA 2023.3.3 2.使用介绍 2.1.启动需要检…

用于图像识别的判别图正则化技术

&#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f916;编程探索专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2024年11月8日13点32分 点击开启你的论文编程之旅https://www.aspiringcode.com/content?id17210272021224&uid64a84f9640714755a…

Android Handler

Handler用于多线程消息分发和处理。与handler相关的几个对象&#xff1a;Message, Looper&#xff0c;MessageQueue, ThreadLocal. Handler是Message的消费者。 MessageQueue是容器。 Looper是整个Message分发的驱动。 Handler中有多种发送消息的方法&#xff0c;其中postxx…

Windows/Linux部署Qt并通过Qt Installer Framework制作安装包

本文参考 Qt Installer Framework Manual。 若要使用 Qt Installer Framework&#xff08;简称 QIF&#xff09;&#xff0c;需要在 Qt Online Installer 或 Qt Maintenance Tool 中确保该组件已安装&#xff08;QIF 组件在 Qt->Developer and Design Tools 下&#xff09;&…