面试 Java 并发编程八股文十问十答第十五期

面试 Java 并发编程八股文十问十答第十五期

作者:程序员小白条,个人博客

相信看了本文后,对你的面试是有一定帮助的!关注专栏后就能收到持续更新!

⭐点赞⭐收藏⭐不迷路!⭐

1)什么是锁的自适应自旋?

在多线程编程中,当一个线程试图获取一个被其他线程持有的锁时,它可能会发生阻塞,等待锁被释放。在一些情况下,等待锁的时间很短,此时采用自旋等待可能更有效率。自旋等待是指线程在尝试获取锁时不会立即进入阻塞状态,而是会反复检查锁是否可用,这样可以减少线程从用户态到内核态的切换开销。

自适应自旋是一种锁优化技术,它根据锁的历史获取情况动态地调整自旋等待的策略。如果锁的获取频率较低,那么自旋等待的时间可以适当延长,以减少线程阻塞和唤醒的开销;反之,如果锁的获取频率较高,那么自旋等待的时间应该相对较短,以减少不必要的自旋消耗。

2)锁如何优化?

锁的优化是提高多线程程序性能的重要手段之一。锁的优化技术可以分为减少锁粒度、减少锁持有时间、锁分离、锁消除等多种方法。具体来说:

  • 减少锁粒度:尽量将锁的范围缩小到最小,避免在不同线程间共享同一个锁,这样可以减少锁竞争的可能性,提高并发性能。
  • 减少锁持有时间:尽量缩短线程持有锁的时间,以减少其他线程等待锁的时间。这可以通过将耗时的操作移到锁外部来实现,或者使用读写锁等更细粒度的锁。
  • 锁分离:将多个互不相关的资源分别加锁,避免一个锁影响其他资源的并发访问。
  • 锁消除:在编译器层面检测到某些锁不可能存在竞争的情况下,可以将这些锁消除掉,以减少锁的开销。
  • 自适应自旋:根据锁的使用情况动态地调整自旋等待的策略,以减少不必要的线程阻塞和唤醒开销。

3)ReentrantLock 的原理?

ReentrantLock 是 Java 中 java.util.concurrent 包提供的一种可重入锁实现。它的原理主要基于 Java 中的 Lock 接口,使用了底层的 AbstractQueuedSynchronizer (AQS) 来管理锁的获取与释放。

ReentrantLock 支持可重入特性,即同一个线程可以多次获取同一把锁,而不会造成死锁。这是通过维护一个持有锁的线程和持有计数的机制来实现的。每个线程持有一个 ThreadLocal 变量,记录了当前线程持有锁的次数。当一个线程多次获取锁时,只需简单地增加持有计数即可。在释放锁时,持有计数减少,直到为零时锁才被完全释放。

ReentrantLock 的底层是基于 AQS 的同步器实现的。AQS 使用一个 state 变量表示同步状态,当 state 为 0 时表示没有线程持有锁,当 state 大于 0 时表示有线程持有锁。通过 state 的状态来实现线程的阻塞和唤醒,以及锁的获取和释放操作。ReentrantLockAQS 的基础上实现了可重入特性,并提供了更多的功能,如公平锁和非公平锁的支持,以及可中断锁等。

4)说说 AQS 吧?

AQS(AbstractQueuedSynchronizer)是 Java 中用于构建锁和其他同步器的基础框架。它提供了一种简单且强大的机制来构建同步器,使得开发者可以实现各种自定义的同步器,比如互斥锁、信号量、倒计时器等。AQS 的核心思想是基于队列的同步器,通过一个先进先出的等待队列来管理线程的排队和唤醒。

AQS 主要包含两种类型的方法:

  • 独占模式方法:用于实现独占锁,即只有一个线程能够获取到锁。这些方法包括 acquire()release() 等。
  • 共享模式方法:用于实现共享锁,即多个线程可以同时获取到锁。这些方法包括 acquireShared()releaseShared() 等。

AQS 的核心数据结构是一个整型的 state 变量,用于表示同步状态。通过操作 state 变量来实现线程的排队和唤醒,以及锁的获取和释放。

AQS 的实现基于模板方法模式,开发者需要继承 AbstractQueuedSynchronizer 类,并实现特定的方法来定义自己的同步器。常见的使用场景包括 ReentrantLockSemaphoreCountDownLatch 等。

5)读写锁知道不?

读写锁是一种特殊的锁机制,允许多个线程同时读取共享资源,但在写操作时会阻塞其他线程的读和写操作,以确保数据的一致性。读写锁分为读锁和写锁两种模式:

  • 读锁(共享锁):多个线程可以同时获取读锁,并发读取共享资源,不会阻塞其他读取操作。但是如果有线程持有写锁,则其他线程无法获取读锁,防止读操作与写操作发生冲突。
  • 写锁(独占锁):写锁是独占的,只有一个线程可以获取写锁,其他线程无法获取读锁或写锁。当有线程持有写锁时,其他线程无法进行读操作或写操作,直到写锁被释放。

读写锁适用于读操作频繁、写操作较少的场景,通过允许多个线程同时读取共享资源,可以提高系统的并发性能。

Java 中提供了 ReentrantReadWriteLock 类来实现读写锁,它是 ReentrantLock 的扩展,同时支持读锁和写锁的功能。

6)CAS 知道不?

CAS(Compare and Swap)是一种原子操作,用于实现多线程环境下的非阻塞算法。它主要用于解决并发环境下的共享数据的更新操作,比如自旋锁、无锁数据结构等。

CAS 操作包括三个参数:要操作的内存位置(通常是一个变量的内存地址)、预期值(即当前内存位置的值)、新值。它的执行过程是先比较内存位置的值是否等于预期值,如果相等,则将内存位置的值更新为新值;如果不相等,则不做任何操作。整个过程是原子的,即在任何时刻只有一个线程能够成功执行 CAS 操作。

CAS 操作通常结合循环重试来使用,如果 CAS 操作失败(即内存位置的值不等于预期值),则重试整个 CAS 操作,直到成功为止。这种方式避免了传统锁机制中的阻塞和唤醒操作,提高了并发性能。

Java 中的 java.util.concurrent.atomic 包提供了一系列原子类,如 AtomicIntegerAtomicLong 等,底层使用了 CAS 操作来实现线程安全的原子操作。

7)什么是 JMM ?

JMM(Java Memory Model)是 Java 内存模型的简称。它定义了 Java 程序中多线程之间的内存访问规则,确保多线程环境下的内存访问操作能够正确地进行同步和交互。

JMM 主要关注以下几个方面:

  • 线程之间的可见性:确保一个线程对共享变量的修改对其他线程可见。
  • 操作的有序性:指令在执行过程中的顺序保证。
  • 原子性:确保一个操作在执行过程中不会被中断,要么全部执行完成,要么完全不执行。

JMM 通过一系列规则和约束来定义这些行为,以保证多线程程序的正确性和可靠性。

8)说说原子性、可见性、有序性?

  • 原子性:指一个操作是不可分割的,要么全部执行成功,要么全部不执行,不存在中间状态。比如对一个整型变量的赋值操作,要么完全执行成功,要么完全不执行。
  • 可见性:指当一个线程修改了共享变量的值后,其他线程能够立即看到这个修改。在多核处理器或多处理器系统中,每个线程有自己的缓存,可见性保证了一个线程对共享变量的修改对其他线程是可见的。
  • 有序性:指程序执行的顺序与代码编写的顺序一致,或者与预期的顺序一致。在多线程环境中,由于指令重排等因素,可能会导致代码的执行顺序与预期不符,有序性保证了指令的执行顺序符合预期。

9)happens-before 听过吗?

“happens-before” 是 Java 内存模型中的概念,用于描述两个操作之间的执行顺序关系。如果一个操作 “happens-before” 另一个操作,则前一个操作的执行结果对后一个操作是可见的。

“happens-before” 关系的几种情况包括:

  • 一个线程中的每个操作 “happens-before” 于该线程中的任何后续操作。
  • 对一个 volatile 字段的写操作 “happens-before” 于后续对该字段的读操作。
  • Monitor 锁的解锁 “happens-before” 于后续对同一锁的加锁。

这种关系保证了程序的有序性和可见性,确保多线程环境下的正确性。

10)指令重排知道吗?

指令重排是指处理器或编译器为了提高指令执行效率而对指令的执行顺序进行优化调整的一种技术。在单线程环境下,指令重排不会对程序的执行结果产生影响,因为最终的执行结果是一致的。但在多线程环境下,指令重排可能会导致程序出现意外结果或者异常行为。

指令重排可能会影响到程序的可见性和有序性,因为它可能会改变代码中操作的执行顺序。为了解决这个问题,Java 内存模型提供了 “happens-before” 的概念,用于约束指令重排的执行顺序,保证多线程环境下的程序行为符合预期。

开源项目地址:https://gitee.com/falle22222n-leaves/vue_-book-manage-system

前后端总计已经 1300+ Star,2 W+ 访问!

⭐点赞⭐收藏⭐不迷路!⭐

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

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

相关文章

Kubernetes 控制平面的安全管理

目录 1. API Server 安全2. etcd 安全3. 网络策略4. 日志与审计5. 定期安全检查与更新6. 云提供商安全集成 Kubernetes 控制平面的安全管理是维护整个集群稳定性和保护敏感信息的关键。控制平面主要包括 API Server、etcd、Controller Manager 和 Scheduler 组件。 1. API Ser…

高斯-牛顿法C实现

高斯-牛顿法(Gauss-Newton method)是一种用于求解非线性最小二乘问题的迭代优化算法。其核心思想是通过近似二阶泰勒展开来简化求解过程,并利用雅可比矩阵(Jacobian matrix)来更新迭代方向。 下面是一个高斯-牛顿法的简单C语言实现示例。这个示例假定我们有一个非线性最小…

Python模块之Numpy(一)-- 创建数组

Numpy是Python用于数据科学计算的基础模块,NumPy 的数据容器能够保存任意类型的数据,可以无缝快速整合各种数据,有助于更加高效地使用pandas等数据处理工具。 数组操作 以下代码是创建一维数组与多维数组并查看数组属性的过程: i…

Mujoco仿真【将urdf文件转化为xml文件】

最近开始学习mujoco仿真方面的内容 先前写过一篇博客:强化学习:MuJoCo机器人强化学习仿真入门(1)_mujoco仿真-CSDN博客 简单介绍了mujoco仿真的一些内容,下面想在Mujoco中将urdf转为xml文件,了解到mujoco是…

Unity值类型和引用类型

我们都知道C#编程语言中,数据类型被分为了两种: 值类型引用类型 那么什么是值类型?什么是引用类型呢?它们的区别又是什么? 为了搞清楚这些问题,我们先列举一下我们开发中会碰到的值类型和引用类型。 常…

【AI+老照片焕新】母亲节用AI把时间的印记变成暖心礼物

想念是一张泛黄的照片,藏在抽屉里的笑容,总是那么亲切。今天是母亲节,是不是想给妈妈来点不一样的惊喜?用AI技术,把那些老照片瞬间焕新,让妈妈的青春记忆重放光华! 想象一下,妈妈年…

使用Giskard进行LLM的测试

Giskard是一个对AI模型进行测试的平台,可以执行功能验证、安全测试及合规扫描。工具主要分为两大块:Giskard Python库和一个server端Giskard Hub。其中Python库是开源的,github地址:https://github.com/Giskard-AI/giskard 使用G…

(delphi11最新学习资料) Object Pascal 学习笔记---第11章第2节 (从接口引用中提取对象)

11.2.5 从接口引用中提取对象 ​ 在过去多个Object Pascal 语言版本中,当你将一个对象赋值给一个接口变量时,是无法访问原始对象的。有时,开发人员会在接口中添加一个 GetObject 方法来执行该操作,但这种设计非常奇怪。 ​ 在今…

华为校招机试 - 模拟汇编计算(20240508)

题目描述 要求设计一种虚拟机解释器,能解析并执行以下虚拟指令。 虚拟机约定: 32位的整型寄存器有 a0,a1,... ,a31 共 32 个寄存器整个虚拟机只有寄存器和立即数参与计算规则集: dst一定为寄存器src为寄存器或十进制正整数运算结果存在负数场景序号虚拟指令含义1MOV dst…

Unity读书系列《Unity高级编程:主程手记》——C#技术要点

文章目录 前言一、业务逻辑优化技巧二、Unity3d中C#的底层原理三、List底层源码剖析四、Dictionary底层源码剖析五、浮点数的精度问题六、委托、事件、装箱、拆箱七、算法总结 前言 本文旨在总结某一概念的性质,并引出相关的技术要点。如果读者希望深入了解相关技术…

docker部署调度程序

Dockerfile(构建初始镜像) # python:3.8-slim-buster为精简版的python FROM python:3.8-slim-buster # 1059为组的id,newgroup为组名,1088为用户的id,newuser为新用户 RUN groupadd -g 1059 newgroup && \useradd -g -u 1088 -g newgroup -m newuser USER newuser RUN…

Python函数和代码复用-课堂练习[python123题库]

函数和代码复用-课堂练习 1、来自计算机的问候-无参无返回值函数 类型:函数‪‬‪‬‪‬‪‬‪‬‮‬‪‬‫‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‪‬‪‬‪‬‪‬‪‬‪‬‮‬‫‬‮‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‮‬‪‬‪‬‪‬‪‬‪‬‮‬‪‬‪‬‪‬‪‬‪‬‪…

react 屏幕信息滚动

1. 引入组件 import DoScrollingbar from @/components/basic/DoScrollingbar; 2.使用组件 <DoScrollingbar>{flightInfo.historytoday && flightInfo.historytoday.map(item => {return <div><div><span style={{ paddingRight: "10px&q…

如何根据招聘信息打造完美简历

如何根据招聘信息打造完美简历 招聘信息分析简历调整策略个性化与关键词结语 在求职过程中&#xff0c;简历是第一块敲门砖。它不仅展示了你的专业技能和工作经验&#xff0c;还体现了你对所申请职位的理解和热情。然而&#xff0c;如何从招聘信息中提炼关键点&#xff0c;打造…

如何在 Ubuntu 上创建一个 SSH CA 以验证主机和客户端

简介 在配置大量具有许多用户的服务器时&#xff0c;保持基础架构的 SSH 访问一致可能会变得复杂。有许多实现集中式身份验证机构的方法&#xff0c;例如 LDAP&#xff0c;但有时这些方法可能过于复杂。 SSH 实际上具有使用证书机构对服务器和客户端进行身份验证的功能。这种…

chatGLM或chatgpt:什么是tokens以及如何计算tokens长度?

token是什么? 简单的来说tokens就是大语言模型输入的向量数据,它是从原始的文本转化而来。 比如 输入:here is a text demo tokens为:[64790, 64792, 985, 323, 260, 2254, 16948] 解码:将tokens转化为文本 [‘[gMASK]’, ‘sop’, ‘▁here’, ‘▁is’, ‘▁a’, ‘▁…

mysql --- 内置函数

MySQL数据库提供了一系列内置函数&#xff0c;这些函数可以在SQL查询中使用&#xff0c;以执行计算、格式化数据、处理字符串、日期和时间等操作。这些函数可以大致分为几个类别&#xff1a;字符串函数、数值函数、日期和时间函数、控制流程函数、加密和压缩函数等。下面是一些…

神经网络模型与前向传播函数

1.概念 在神经网络中&#xff0c;模型和前向传播函数是紧密相关的概念。模型定义了网络的结构&#xff0c;而前向传播函数描述了数据通过网络的流动方式。以下是这两个概念的详细解释&#xff1a; 1.1 神经网络模型 神经网络模型是指构成神经网络的层、权重、偏置和连接的集…

SpringBoot拦截器中使用RedisTemplate

这几天想着把登陆拦截器的验证规则修改一下&#xff0c;验证介质由session中获取改为从redis中获取&#xff0c;结果发现redisTemplate一直为空&#xff0c; Configuration public class WebInterceptorConfig implements WebMvcConfigurer {Overridepublic void addIntercept…

源码知识付费系统,在线教学平台需要优化什么?

在线教育关于广大的关注者而言属于快捷度非常高的传达途径&#xff0c;尤其是白日没有过多时间的上班族或学习繁忙的学生&#xff0c;均能够通过可靠的在线教育完结自己的目的。如此巨大的市场潜力使得以在线教育为主的公司数量呈现出直线上升的趋势&#xff0c;很多的在线教育…