Java面试题之线程通信的方式

简介: 一、线程通信的方式二、线程协作-JDK API1、被弃用的suspend和resume(1) 正常的suspend/resume使用示例(2) 造成死锁的suspend/resume示例2、wait/notify机制(1) wait/notify正确使用示例(2) wait/notify死锁示例(notify先调用)3、park/unpark机制(1) 正确的park/unpark使用示例(2) 死锁的park/unpark三、伪唤醒

一、线程通信的方式

二、线程协作-JDK API

1、被弃用的suspend和resume

(1) 正常的suspend/resume使用示例

(2) 造成死锁的suspend/resume示例

2、wait/notify机制

(1) wait/notify正确使用示例

(2) wait/notify死锁示例(notify先调用)

3、park/unpark机制

(1) 正确的park/unpark使用示例

(2) 死锁的park/unpark

三、伪唤醒

一、线程通信的方式



要实现多个线程之间的协同,如:线程执行先后顺序、获取某个线程的执行结果等等。线程之间相互通信,分为下面四类:

  • 文件共享。
  • 网络共享。
  • 共享变量。
  • jdk提供的线程协调api,有suspend/resume、wait/notify、park/unpark。

二、线程协作-JDK API



、被弃用的suspend和resume

(1) 正常的suspend/resume使用示例

作用:调用suspend挂起目标线程,通过resume可以恢复线程。c8697aad1e474e6fa03d96e814d8ff8e.png

(2) 造成死锁的suspend/resume示例

被弃用的主要原因:容易写出死锁的代码,所以用wait/notify、park/unpark机制对它就行替代,resume先调用造成死锁:5c0a788d0da94cc3ac8784ac4aa2b454.png2、wait/notify机制

wait/notify机制相关介绍:


这些方法只能只能由同一对象锁的持有者线程调用,也就是写在同步代码块里面,否则会抛出IllegalMonitorException。

wait方法导致当前线程等待,加入该对象的等待集合中,并放弃当前线程持有的锁。

notify/notifyAll方法唤醒一个或所有正在等在这个对象锁的线程。

虽然wait会自动解锁,但是对顺序有要求,如果在notify被调用之后,才开始wait方法的调用,线程会处于waiting状态。

(1) wait/notify正确使用示例

169868f29cac4d70a4f9758a4a1f53a7.png

(2) wait/notify死锁示例(notify先调用)b4eceaaf6d5a49b4baf747dc618e4f0a.png

3、park/unpark机制

线程调用“park”则等待许可,unpark方法为指定线程提供“许可”。

不要求park和unpark方法的调用顺序。

多次调用unpark之后,再调用park,线程会直接运行。但不会叠加,也就是说,连续多次调用park方法,第一次会拿到“许可”直接运行,后续调用会进入等待。

(1) 正确的park/unpark使用示例

99d93dd489d2479b983a6112a82e846f.png

(2) 死锁的park/unparke29fd35bd28e48979a884d907dd9ee67.png

三、伪唤醒


  • 官方建议在循环中检查等待条件,原因是处于等待状态的线程可能会收到错误警报和伪唤醒,如果不在循环中检查等待条件,程序在没有满足结束条件的情况下会退出。
  • 伪唤醒是指线程并非因为notify、notifyAll、unpark等api调用唤醒,是更底层原因导致的。d51423e97b9b4d5c8563c084379baa75.png
相关文章
|
7天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
25 2
|
12天前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
18天前
|
存储 缓存 Oracle
Java I/O流面试之道
NIO的出现在于提高IO的速度,它相比传统的输入/输出流速度更快。NIO通过管道Channel和缓冲器Buffer来处理数据,可以把管道当成一个矿藏,缓冲器就是矿藏里的卡车。程序通过管道里的缓冲器进行数据交互,而不直接处理数据。程序要么从缓冲器获取数据,要么输入数据到缓冲器。
Java I/O流面试之道
|
26天前
|
Java 调度
[Java]线程生命周期与线程通信
本文详细探讨了线程生命周期与线程通信。文章首先分析了线程的五个基本状态及其转换过程,结合JDK1.8版本的特点进行了深入讲解。接着,通过多个实例介绍了线程通信的几种实现方式,包括使用`volatile`关键字、`Object`类的`wait()`和`notify()`方法、`CountDownLatch`、`ReentrantLock`结合`Condition`以及`LockSupport`等工具。全文旨在帮助读者理解线程管理的核心概念和技术细节。
37 1
[Java]线程生命周期与线程通信
|
12天前
|
Java
JAVA多线程通信:为何wait()与notify()如此重要?
在Java多线程编程中,`wait()` 和 `notify()/notifyAll()` 方法是实现线程间通信的核心机制。它们通过基于锁的方式,使线程在条件不满足时进入休眠状态,并在条件满足时被唤醒,从而确保数据一致性和同步。相比其他通信方式,如忙等待,这些方法更高效灵活。 示例代码展示了如何在生产者-消费者模型中使用这些方法实现线程间的协调和同步。
26 3
|
14天前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
39 4
|
15天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
55 4
|
26天前
|
存储 Java
[Java]面试官:你对异常处理了解多少,例如,finally中可以有return吗?
本文介绍了Java中`try...catch...finally`语句的使用细节及返回值问题,并探讨了JDK1.7引入的`try...with...resources`新特性,强调了异常处理机制及资源自动关闭的优势。
20 1
|
25天前
|
算法 Java
JAVA 二叉树面试题
JAVA 二叉树面试题
14 0
|
1月前
|
存储 消息中间件 资源调度
C++ 多线程之初识多线程
这篇文章介绍了C++多线程的基本概念,包括进程和线程的定义、并发的实现方式,以及如何在C++中创建和管理线程,包括使用`std::thread`库、线程的join和detach方法,并通过示例代码展示了如何创建和使用多线程。
44 1
C++ 多线程之初识多线程
下一篇
无影云桌面