面试 Java 并发编程八股文十问十答第四期
面试 Java 并发编程八股文十问十答第四期
作者:程序员小白条,个人博客
相信看了本文后,对你的面试是有一定帮助的!关注专栏后就能收到持续更新!
⭐点赞⭐收藏⭐不迷路!⭐
1)Java 中你怎样唤醒一个阻塞的线程?
要唤醒一个阻塞的线程,通常需要使用wait()、notify() 和 notifyAll() 方法配合使用。下面是一般的步骤:
- 在同步代码块中调用
wait()方法使线程进入等待状态。 - 当某个条件满足时,调用
notify()或notifyAll()方法唤醒等待的线程。 - 被唤醒的线程会重新竞争对象锁,一旦获得锁,就可以继续执行。
synchronized (monitor) {while (condition) {monitor.wait(); // 线程进入等待状态}// 满足条件后唤醒线程monitor.notify(); // 或者 monitor.notifyAll();
}
2)notify() 和 notifyAll() 有什么区别?
notify()方法用于唤醒等待队列中的一个线程,具体唤醒哪个线程取决于调度器的选择。notifyAll()方法用于唤醒等待队列中的所有线程,让它们有机会竞争对象锁。
通常情况下,如果只有一个线程需要被唤醒,可以使用 notify() 方法以减少不必要的竞争。但是,如果有多个线程需要被唤醒,或者不确定有多少线程需要唤醒时,可以使用 notifyAll() 方法确保所有等待的线程都有机会继续执行。
3)如何在两个线程间共享数据?
在 Java 中,两个线程之间共享数据的常见方式包括使用共享对象或者使用线程间通信机制。以下是一些常见的方法:
- 共享对象:两个线程可以共享同一个对象,通过对象的成员变量来共享数据。需要确保线程安全,可以使用同步方法或同步块来保护共享数据的访问。
- 线程间通信:可以使用
wait()、notify()、notifyAll()等方法来实现线程间的通信。一个线程在等待某个条件满足时调用wait()进入等待状态,另一个线程在满足条件时调用notify()或notifyAll()唤醒等待的线程。 - 线程安全的数据结构:使用线程安全的数据结构,比如
ConcurrentHashMap、ConcurrentLinkedQueue等,来实现线程间的数据共享。 - 使用锁:可以使用
ReentrantLock或synchronized关键字来保护共享数据的访问,确保线程安全。
在实现线程间数据共享时,需要注意线程安全性和避免发生竞态条件,以确保多线程环境下的数据一致性和正确性。
4)Java 如何实现多线程之间的通讯和协作?
在 Java 中,多线程之间的通讯和协作可以通过以下方式实现:
- 使用共享对象:多个线程共享同一个对象,通过对象的成员变量进行通讯。可以使用
wait()、notify()和notifyAll()方法实现线程间的等待和唤醒。 - 使用线程安全的队列:可以使用
BlockingQueue或其他线程安全的数据结构来实现线程之间的数据传递和协作。 - 使用信号量:
Semaphore类可以用来控制同时访问共享资源的线程数量,实现线程之间的协作。 - 使用 CountDownLatch 和 CyclicBarrier:
CountDownLatch和CyclicBarrier类可以用来实现多个线程之间的协作和同步。 - 使用 Lock 和 Condition:
ReentrantLock和Condition类可以实现更灵活的线程协作机制。
5)同步方法和同步块,哪个是更好的选择?
在选择同步方法和同步块时,取决于具体情况。一般来说:
- 同步方法:使用同步方法可以简化代码,因为整个方法体都会被同步,不需要显式地编写同步代码块。但是同步方法的粒度比较粗,可能会影响程序的性能。
- 同步块:使用同步块可以精确地控制需要同步的代码块,避免不必要的同步,提高程序的性能。但是需要显式地编写同步代码块,增加了代码的复杂性。
一般来说,如果需要同步的代码比较少,可以考虑使用同步块;如果整个方法都需要同步,可以考虑使用同步方法。在实际应用中,需要根据具体情况进行权衡和选择。
6)什么是线程同步和线程互斥,有哪几种实现方式?
- 线程同步:线程同步是指多个线程之间按照一定的顺序执行,保证共享资源的正确访问。线程同步可以避免竞态条件和数据不一致的问题。
- 线程互斥:线程互斥是指多个线程在访问共享资源时,通过加锁的方式实现互斥访问,即同一时刻只有一个线程可以访问共享资源,其他线程需要等待。
几种实现线程同步和线程互斥的方式包括:
- 使用 synchronized 关键字:可以在方法上或代码块中使用
synchronized关键字来实现线程同步和互斥。 - 使用 Lock 接口:
ReentrantLock类提供了与synchronized关键字类似的功能,可以更灵活地控制锁的获取和释放。 - 使用 volatile 关键字:
volatile关键字可以保证变量的可见性,但不能保证原子性,适用于简单的标志位等情况。 - 使用 wait()、notify() 和 notifyAll():通过这些方法可以实现线程之间的等待和唤醒,实现线程间的协作和互斥。
- 使用并发容器:Java 中的并发容器如
ConcurrentHashMap、ConcurrentLinkedQueue等提供了线程安全的数据结构,可以避免手动加锁。
选择合适的线程同步和互斥方式取决于具体的需求和场景,需要根据实际情况进行选择和权衡。
7)在监视器(Monitor)内部,是如何做线程同步的?程序应该做哪种级别的同步?
在 Java 中,监视器(Monitor)是通过对象的内置锁(也称为监视器锁)来实现线程同步的。当一个线程进入一个 synchronized 方法或代码块时,它会尝试获取对象的内置锁,如果锁已经被其他线程持有,则该线程会被阻塞,直到锁被释放。
程序应该根据需要选择合适的同步级别,可以在方法级别或代码块级别进行同步。一般来说,推荐使用更细粒度的同步,即在尽可能小的代码块中加锁,以减少同步的范围,提高程序的性能。
8)如果你提交任务时,线程池队列已满,这时会发生什么?
当线程池队列已满时,如果线程池中的线程数量未达到最大线程数限制,新提交的任务会尝试创建新的线程来执行。如果线程池中的线程数量已经达到最大线程数限制,且队列已满,则根据线程池的拒绝策略来处理新提交的任务。
常见的拒绝策略包括:
- AbortPolicy:直接抛出 RejectedExecutionException 异常。
- CallerRunsPolicy:由提交任务的线程来执行这个任务。
- DiscardPolicy:直接丢弃新任务,不抛出异常。
- DiscardOldestPolicy:丢弃队列中等待时间最长的任务,然后尝试重新提交新任务。
可以根据具体的业务需求选择合适的拒绝策略,或者自定义拒绝策略来处理任务提交时线程池队列已满的情况。
9)什么叫线程安全?servlet 是线程安全吗?
线程安全指的是在多线程环境下,对共享数据的访问操作不会导致数据出现异常或不一致的情况。线程安全的代码能够正确地处理多个线程并发访问共享数据的情况,保证数据的一致性和正确性。
对于 Servlet,Servlet 容器会为每个请求创建一个线程来处理,因此 Servlet 的实现需要是线程安全的。Servlet 容器会在多个线程之间共享同一个 Servlet 实例,因此 Servlet 类中的实例变量需要考虑线程安全性。
通常情况下,Servlet 应该遵循以下原则来确保线程安全:
- 避免使用实例变量存储请求相关的数据,可以使用局部变量或者 synchronized 来处理共享数据。
- 如果需要使用实例变量,确保对共享数据的访问是线程安全的,可以使用 synchronized 或其他线程安全的方式来保护数据访问。
10)在 Java 程序中怎么保证多线程的运行安全?
要保证多线程的运行安全,可以采取以下措施:
- 使用同步机制:使用 synchronized 关键字或 Lock 接口来保护共享数据的访问,确保在同一时刻只有一个线程可以访问共享资源。
- 使用线程安全的数据结构:使用 Java 提供的线程安全的数据结构,如 ConcurrentHashMap、ConcurrentLinkedQueue 等,避免手动加锁。
- 避免竞态条件:在编写多线程程序时,要注意避免竞态条件的发生,即多个线程同时对共享资源进行读写而导致数据不一致的情况。
- 使用 volatile 关键字:在需要保证变量可见性的情况下,可以使用 volatile 关键字来修饰变量,确保线程之间的数据同步。
- 使用线程池:合理使用线程池可以避免线程创建和销毁的开销,同时可以控制并发线程数量,提高程序的性能和稳定性。
- 避免死锁:在设计多线程程序时,要避免出现死锁情况,即多个线程相互等待对方释放资源而无法继续执行的情况。
通过以上措施,可以有效地保证多线程程序的运行安全,确保数据的一致性和正确性。
开源项目地址:https://gitee.com/falle22222n-leaves/vue_-book-manage-system
前后端总计已经 800+ Star,1.5W+ 访问!
⭐点赞⭐收藏⭐不迷路!⭐
相关文章:
面试 Java 并发编程八股文十问十答第四期
面试 Java 并发编程八股文十问十答第四期 作者:程序员小白条,个人博客 相信看了本文后,对你的面试是有一定帮助的!关注专栏后就能收到持续更新! ⭐点赞⭐收藏⭐不迷路!⭐ 1)Java 中你怎样唤醒…...
物体检测-系列教程27:YOLOV5 源码解析17(训练脚本解读:训练函数4)
😎😎😎物体检测-系列教程 总目录 有任何问题欢迎在下面留言 本篇文章的代码运行界面均在Pycharm中进行 本篇文章配套的代码资源已经上传 点我下载源码 24、epoch循环训练------更新、评估、保存 这部分是训练过程的每个epoch结束之前执行的一…...
基于51单片机的数字时钟(万年历)设计与实现
基于51单片机的数字时钟(万年历)设计与实现 摘要 随着科技的不断发展,数字时钟已成为人们日常生活中不可或缺的一部分。基于51单片机的数字时钟(万年历)设计,结合了传统时钟的功能与现代电子技术…...
2024年谷歌SEO的趋势预测及应对建议(川圣SEO)蜘蛛池
baidu搜索:如何联系八爪鱼SEO? baidu搜索:如何联系八爪鱼SEO? baidu搜索:如何联系八爪鱼SEO? 虽然说“SEO”已死这个口号已经喊了很多年了(最终也没死),但是在2023年很…...
Rust 生命周期符号使用的方法和规律
一、生命周期符号使用的规律 在 Rust 中,生命周期(lifetimes)是用于处理引用和所有权问题的关键概念。生命周期符号(通常表示为 a、b 等)用于指定引用的有效时间范围。这有助于编译器确保在引用被使用时,所…...
生成哈夫曼树(100%用例)C卷(JavaPythonC++Node.jsC语言)
给定长度为n的无序的数字数组,每个数字代表二叉树的叶子节点的权值,数字数组的值均大于等于1。请完成一个函数,根据输入的数字数组,生成哈夫曼树,并将哈夫曼树按照中序遍历输出。 为了保证输出的二又树中序遍历结果统一,增加以下限制:二叉树节点中,左节点权值小于等于右…...
el-form-item内的el-select如何自适应宽度
最近在使用element-ui做后台管理的时候,有个需求是在弹窗组件里面,添加一个el-select下拉框选项,但是给el-select设置的宽度无法自适应,原因很简单,我们不需要设置固定宽度,设置百分比就行了,让…...
什么洗地机值得推荐?旗舰洗地机希亦、追觅、西屋、海尔实际表现如何?
洗地机这个产品相信大家已经不陌生了,它集合吸尘器和电动扫地拖把的功能,轻轻推拉便可以解决地面上的赃物,且不用我们手动清洗滚刷,深得家务人的喜爱,可是,当我们真正要去选购的时候,还是很纠结…...
掘根宝典之C++隐式类型转化(整型提升,算术转换)
赋值中的隐式转换 话不多说,我们直接看例子 bool b42; //b为真 int ib; //i的值为1 i3.14; //i的值为3 double pi i; // pi的值为3.0 unsigned char c -1; // 假设char占8比特,c的值为255 s…...
group by order by having where union
力扣题目链接 having where 区别 having子句用于分组后筛选,where子句用于行条件筛选 having一般都是配合group by 和聚合函数一起出现如(count(),sum(),avg(),max(),min()) where条件子句中不能使用聚集函数,而having子句就可以。 having只能用在group…...
微信私信短剧机器人源码
本源码仅提供参考,有能力的继续开发 接口为api调用 云端同步 https://ys.110t.cn/api/ajax.php?actyingshilist 影视搜索 https://ys.110t.cn/api/ajax.php?actsearch&name剧名 每日更新 https://ys.110t.cn/api/ajax.php?actDaily 反馈接口 https://ys.11…...
java使用ws.schild.jave将视频转成mp4
<?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation"http://maven.apache.org/POM/4.0.0 http://…...
python map函数
python map函数 文章目录 python map函数 在Python中, map()函数用于将一个函数应用于可迭代对象(如列表或元组)中的每个元素,并返回一个包含结果的新的可迭代对象。 map()函数的语法如下: map(function, iterable)其…...
基于SSM的党务政务服务热线平台(有报告)。Javaee项目。ssm项目。
演示视频: 基于SSM的党务政务服务热线平台(有报告)。Javaee项目。ssm项目。 项目介绍: 采用M(model)V(view)C(controller)三层体系结构,通过Spri…...
Unity3D 动态生成场景管理节点详解
前言 Unity3D 提供了丰富的功能和工具,可以帮助开发者快速高效地创建各种类型的游戏。在游戏开发过程中,有时候我们需要动态生成场景管理节点来管理游戏场景中的各种元素,比如角色、道具、敌人等。本文将详细介绍如何在Unity3D中动态生成场景…...
js--构造函数
创建对象的方式: 1、利用对象字面量{}创建 const arr {name: tom,age: 18 } 2、利用js内置构造(Object,Array,String,Number)函数 var obj new Object() //创建一个空的对象 obj.uname tom obj.age 2…...
Tomcat目录结构
文章目录 binconfliblogswebapp bin 存放tomcat的可执行程序 从上图可以看出bin中的文件主要是两种文件,一种是.bat一种是.sh .bat:主要用于windows .sh:主要用于linux .bat文件是Windows操作系统中的批处理文件。它是一种简单的文本文件,其中包含了一…...
读西游记第一回:西游记世界格局
天地之数: 元:十二万九千六百岁(129600年) 1元12会:子、丑、寅、卯、巳、午、未、申、酉、戌、亥。每会18000年。与12地支对应。 亥会期:前5400年混沌期,后5400年,盘古开天辟地&am…...
【Unity知识点详解】Button点击事件拓展,单击、双击、长按实现
Button拓展 今天来聊一下关于Button的事件拓展,这里只是拿Button来举例,Unity中其他的UI组件如Toggle、Slider等都也适用。 我们知道在Button中我们可以通过onClick的方式来添加点击事件,但在游戏开发过程中我们往往对Button有着更多的功能需…...
了解财富的本质才能知道自己几斤几两
生活在现代都市中,经历了经济的潮起潮落。在一望无际的楼宇下,是每天匆忙工作的一个个鲜活个体。有的在为了生存而工作,有的在享受着惬意的时光,有人行色匆匆,目光所及之处,尽是可遇不可求的机会。成为中产…...
微软PowerBI考试 PL300-选择 Power BI 模型框架【附练习数据】
微软PowerBI考试 PL300-选择 Power BI 模型框架 20 多年来,Microsoft 持续对企业商业智能 (BI) 进行大量投资。 Azure Analysis Services (AAS) 和 SQL Server Analysis Services (SSAS) 基于无数企业使用的成熟的 BI 数据建模技术。 同样的技术也是 Power BI 数据…...
《从零掌握MIPI CSI-2: 协议精解与FPGA摄像头开发实战》-- CSI-2 协议详细解析 (一)
CSI-2 协议详细解析 (一) 1. CSI-2层定义(CSI-2 Layer Definitions) 分层结构 :CSI-2协议分为6层: 物理层(PHY Layer) : 定义电气特性、时钟机制和传输介质(导线&#…...
postgresql|数据库|只读用户的创建和删除(备忘)
CREATE USER read_only WITH PASSWORD 密码 -- 连接到xxx数据库 \c xxx -- 授予对xxx数据库的只读权限 GRANT CONNECT ON DATABASE xxx TO read_only; GRANT USAGE ON SCHEMA public TO read_only; GRANT SELECT ON ALL TABLES IN SCHEMA public TO read_only; GRANT EXECUTE O…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
智能分布式爬虫的数据处理流水线优化:基于深度强化学习的数据质量控制
在数字化浪潮席卷全球的今天,数据已成为企业和研究机构的核心资产。智能分布式爬虫作为高效的数据采集工具,在大规模数据获取中发挥着关键作用。然而,传统的数据处理流水线在面对复杂多变的网络环境和海量异构数据时,常出现数据质…...
Java多线程实现之Thread类深度解析
Java多线程实现之Thread类深度解析 一、多线程基础概念1.1 什么是线程1.2 多线程的优势1.3 Java多线程模型 二、Thread类的基本结构与构造函数2.1 Thread类的继承关系2.2 构造函数 三、创建和启动线程3.1 继承Thread类创建线程3.2 实现Runnable接口创建线程 四、Thread类的核心…...
华硕a豆14 Air香氛版,美学与科技的馨香融合
在快节奏的现代生活中,我们渴望一个能激发创想、愉悦感官的工作与生活伙伴,它不仅是冰冷的科技工具,更能触动我们内心深处的细腻情感。正是在这样的期许下,华硕a豆14 Air香氛版翩然而至,它以一种前所未有的方式&#x…...
JAVA后端开发——多租户
数据隔离是多租户系统中的核心概念,确保一个租户(在这个系统中可能是一个公司或一个独立的客户)的数据对其他租户是不可见的。在 RuoYi 框架(您当前项目所使用的基础框架)中,这通常是通过在数据表中增加一个…...
SQL慢可能是触发了ring buffer
简介 最近在进行 postgresql 性能排查的时候,发现 PG 在某一个时间并行执行的 SQL 变得特别慢。最后通过监控监观察到并行发起得时间 buffers_alloc 就急速上升,且低水位伴随在整个慢 SQL,一直是 buferIO 的等待事件,此时也没有其他会话的争抢。SQL 虽然不是高效 SQL ,但…...
Ubuntu Cursor升级成v1.0
0. 当前版本低 使用当前 Cursor v0.50时 GitHub Copilot Chat 打不开,快捷键也不好用,当看到 Cursor 升级后,还是蛮高兴的 1. 下载 Cursor 下载地址:https://www.cursor.com/cn/downloads 点击下载 Linux (x64) ,…...
