多线程学习笔记(三)

news2025/1/16 10:57:00

一、缓存行

缓存行:从缓存中读取数据是按照一块来读取的,这一块叫做缓存行,64字节大小
缓存一致性协议:当两个数位于同一个缓存行时,有两个线程需要同时读取了缓存行中的数据后进行修改,需要和另一个线程的数据保存一致

可以通过在数据前后补充空数据来保证两个数据不在同一个缓存行,来避免反复进行缓存一致性(jdk1.7中LinkedBlockingQueue就用了此方式)

jdk1.8中,可以通过@Contended来保证数据自己在一个缓存行中,          需要在jvm运行参数中加  -XX:RestrictContended

缓存一致性协议根据不同cpu有不同实现,intel的cpu的缓存一致性协议是MESI

M:Modified

E:Exclusive

S: Shared

I: Invalid 

二、乱序

1. 为什么会出现乱序

为了提高效率

2. 乱序的存在条件

不影响单线程的最终一致性(as if serial)

指令重排可能会导致对象逃逸(不要在构造方法中启动线程)

3. 对象在内存中的存储布局

 对象的大小是8字节的倍数,如果前面三个占用10字节,最后的padding需要加6字节对齐

markword占8字节

class pointer默认压缩,占用4字节

instance data默认压缩

对象头包含markword和class pointer

4. hanppens-before 原则

JVM规定重排序必须要遵循的规则

5. 内存屏障

使用内存屏障阻止乱序执行

内存屏障是特殊指令:看到这种指令,前面的必须执行完,后面的才能执行

intel的cpu的屏障指令:lfence(load 读)、sfence(store 写)、mfence(读写)

6. jvm中的内存屏障

LoadLoadBarrier、LoadStoreBarrier、StoreLoadBarrier、StoreStoreBarrier

7. volatile怎么实现禁止指令重排序

在volatile写操作前面加StoreStoreBarrier,后面加StoreLoadBarrier

在volatile读操作前面加LoadLoadBarrier、后面加LoadStoreBarrier

volatile在hotspot中的实现是通过 lock addl指令

三、原子性

1. 概念

不能被打断的操作,需要作为一个整体执行的操作,叫做原子操作

2. 上锁的本质

把并发编程序列化

3. 实现

通过乐观锁和悲观锁来保证线程的原子性

乐观锁:认为这个操作不会被别的线程打断,使用cas操作

悲观锁:认为这个操作会被别的线程打断,synchronized就是悲观锁

4. CAS:compare and swap

假设要给a + 1,先读取到a,然后把a值+1,再去跟一开始读取的a进行比较,如果此时读取到的a跟原来的一样,就给a替换成新值,否则,重新读取a再进行操作

ABA问题,如果比较的a跟之前一样,但是中间可能已经发生了改变,只是又变回原来的值了

解决ABA问题,加version

CAS操作本身要保证原子性,要不然中间可能会出问题

AtomicInteger可以实现原子性,通过CAS操作,在汇编指令上通过lock cmpxchg实现

5. 选择乐观锁还是悲观锁

临界区执行时间较长,等待线程较多,使用悲观锁

临界区执行时间短,等待线程较少,使用乐观锁

四、synchronized

用户态:只能执行部分指令

内核态:所有指令都可以执行

jvm工作在用户态,synchronized重量级锁要通过操作系统才能申请到 

1. markword

 根据最后两位判断,

如果是00,表示轻量级锁

如果是10,表示重量级锁

如果是01,需要根据倒数第三位来判断,如果倒数第三位是0,表示无锁(刚刚new出来);如果是1,表示偏向锁

偏向锁和轻量级锁是用户空间的完成

重量级锁需要向内核申请

2. 重入次数

synchronized是可重入锁,需要记录重入次数

偏向锁和轻量级锁的重入次数记录在线程栈中,LockRecord+1

重量级锁记录在ObjectMonitor上

3. 锁升级

 

jvm刚启动时,默认不开启偏向锁,默认4秒后开启偏向锁(可以通过参数设置)。

刚启动就开启偏向锁的对象没有偏向,所以就称为匿名偏向。

匿名偏向对象升级的时候升级为偏向锁,偏向锁升级的时候升级为轻量级锁

没有偏向的对象会直接升级为轻量级锁

当第一个来申请锁的对象会把自己的线程id放入markword中,称为偏向锁

当好几个对象都来申请锁的时候,会通过CAS自旋(在线程栈中写入LockRecord)来升级为轻量级锁

jdk1.4中通过自旋次数,占用cpu数来决定升级为重量级锁,1.6及以后,自适应升级重量级锁

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

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

相关文章

如何使用CCES来生成双核DSP的可烧写文件以及Flash Program

作者的话 之前的文章里有写到如何使用CCES来进行DSP的flash program,那都是单核的DSP,而ADI还有很多双核甚至三核DSP,这些DSP应该如何操作呢,会有哪些不同?我再写一个文档来说一下。 跟单核DSP一样,在 CC…

【测绘程序设计】——角度与弧度的转换(C#,C++,Python)

本文分享了 测绘程序设计——角度与弧度的转换(含C#、C++和Python版),相关源代码(完整工程,直接运行)及使用示例如下。 目录 Part.Ⅰ 使用示例Chap.Ⅰ C#版Chap.Ⅱ C++版Chap.Ⅲ Python版Part.Ⅱ 代码分析Chap.Ⅰ C#版Chap.Ⅱ C++版Chap.Ⅲ Python版Part.Ⅲ 源码下载Part…

实战YOLO V5推演(TensorRT版本 C++)

提示:兜兜转转还是来到了C,欠的债该还的得还,因此本篇文章试图从C来撬动tensorrt 完成转换模型和推理,而不是借助python库 文章目录前言一、Tensorrt前置1.1 Tensorrt基本概念1.2 异步推演说明二、实战YOLO V5推演1.TensorRT模型转…

大学电子系C++模拟考试

随手附上一些代码,未必是最优解,仅供参考。 加密四位数 【问题描述】 输入一个四位数,将其加密后输出。方法是将该数每一位的数字加9,然后除以10取余作为该位上的新数字,最后将千位上的数字和十位上的数字互换&#…

Prometheus 监控mysql

目录 下载安装mysqld_exporter 在mysql中创建监控用户并赋权 启动mysqld_exporter 添加到系统服务 浏览器访问服务器9104端口 在prometheus定义job来监控mysqld 运行prometheus并在端口9090上进行访问查看节点信息 根据具体需求再在可视化平台上定义数据源来进行可视化…

Java数据结构之树与二叉树

2.3 二叉树的性质(重点,选择题常考) 2.4 二叉树的链式存储 2.5 二叉树的基本操作 2.5.1 前提说明 2.5.2 二叉树的遍历 2.5.3 二叉树基本操作的实现(重点) 1. 树形结构 1.1 树的概念 树是一种非线性的数据结构&…

[附源码]JAVA毕业设计体检系统(系统+LW)

[附源码]JAVA毕业设计体检系统(系统LW) 项目运行 环境项配置: Jdk1.8 Tomcat8.5 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术&#xf…

JavaScript实现一段时间之后关闭广告

JavaScript实现一段时间之后关闭广告 案例池子: JS实现鼠标悬停变色 JavaScript中的排他算法实现按钮单选 JavaScript中的localStorage JavaScript中的sessionStorage JavaScript实现网页关灯效果 JavaScript实现一段时间之后关闭广告 JavaScript实现按键快速获…

水下机器人双机械手系统动态建模与控制仿真(Matlab代码实现)

目录 💥1 概述 📚2 运行结果 🎉3 参考文献 👨‍💻4 Matlab代码 💥1 概述 水下机器人-机械手系统(Underwater vehicle-manipulator systems, UVMS)可以完成除观测之外的水下采样…

[论文解析] HeadNeRF: A Realtime NeRF-based Parametric Head Model

code link: https://github.com/CrisHY1995/headnerf 相关连接: https://blog.csdn.net/c9Yv2cf9I06K2A9E/article/details/124874717 来自中科大张举勇教授课题组提出了 HeadNeRF,一种基于 NeRF 的高效全息人脸头部参数化模型,该工作发表…

阿里面试Redis最常问的三个问题:缓存穿透、缓存击穿、缓存雪崩(带答案)

一、缓存穿透 如上图,正常情况下,如果用户在redis缓存中没有查询到自己想要的数据,就会去mysql数据库中查询。如果mysql数据库中也没有,在没有任何措施下,用户一定会不断的去mysql数据库中查询,随着时间的推…

风控串行组合模型及其信用评估场景实践

在信贷风控中,数据建模好像是“家常便饭”那么普通且重要,而我们最终享用“饭菜”的美味程度,在数据“食材”较完备的情况下,完全取决于我们建模的方法。根据实际业务场景,采用合理且有效的建模思路,可以较…

诊断和响应故障_RMAN数据修复概念

1.RMAN数据修复概述 1.1.关于需要数据修复的问题 虽然有几种问题会中止Oracle数据库的正常操作或影响数据库I/O操作,只有以下几种情况要求DBA介入和数据修复:用户错误,应用程序错误和介质故障。 1.1.1.关…

[附源码]计算机毕业设计JAVA中达小区物业管理系统

[附源码]计算机毕业设计JAVA中达小区物业管理系统 项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM my…

【LeetCode】1775. 通过最少操作次数使数组的和相等

题目描述 给你两个长度可能不等的整数数组 nums1 和 nums2 。两个数组中的所有值都在 1 到 6 之间(包含 1 和 6)。 每次操作中,你可以选择 任意 数组中的任意一个整数,将它变成 1 到 6 之间 任意 的值(包含 1 和 6&…

LaTeX行距以及字体大小

LaTeX行距以及字体大小1、行距2、字体大小1、全局模式2、局部模式1、行距 一:改变全文行距 导言部分前加入\linespread{2.0}即可,代表全文两倍行距。 二:部分段落需要改变行距 \usepackage{setspace} \begin{spacing}{2.0}(内…

js_实现网页自动跳转

网页自动跳转实现网页定时自动跳转实现网页定时自动跳转 效果&#xff1a; 10秒后自动跳转到网易云音乐的页面 代码&#xff1a; <!DOCTYPE html> <html><head><meta charset"utf-8"><title></title><style>.box{wi…

什么是 NFT 洗盘交易:洗盘交易背后的原理

2022年12月 CoinGecko 与 Footprint Analytics 联合报告 Data Source&#xff1a;Footprint Analytics 市场上有一种说法是&#xff0c;整个NFT 市场的交易本质是欺诈性的&#xff0c;并且是由洗盘交易者操纵出来的假象。我们经常看到许多 媒体头条 在推动这种说法。毕竟&…

HTML学生个人网站作业设计:宠物网站设计——萌宠有家(5页) HTML+CSS 简单DIV布局宠物介绍网页模板代码 DW学生个人网站制作成品下载

&#x1f389;精彩专栏推荐 &#x1f4ad;文末获取联系 ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业&#xff1a; 【&#x1f4da;毕设项目精品实战案例 (10…

动态规划:01背包问题例题(leetcode+cpp实现)

文章目录分割等和子集最后一块石头的重量前情回顾&#xff1a; 动态规划(4)&#xff1a;01背包问题详解 分割等和子集 力扣传送门&#xff1a; https://leetcode.cn/problems/partition-equal-subset-sum/ 题目描述&#xff1a; 给你一个整数数组&#xff0c;将这个数组里的…