Java多线程面试题总结(中)

简介: Java内存模型(JMM)定义了程序中所有变量的访问规则与范围,确保多线程环境下的数据一致性。JMM包含主内存与工作内存的概念,通过8种操作管理两者间的交互,确保原子性、可见性和有序性。`synchronized`和`volatile`关键字提供同步机制,前者确保互斥访问,后者保证变量更新的可见性。多线程操作涉及不同状态,如新建(NEW)、可运行(RUNNABLE)等,并可通过中断、等待和通知等机制协调线程活动。`volatile`虽不确保线程安全,但能确保变量更新对所有线程可见。

::: tip
这个里面的内容对应 Java并发编程基础知识 书籍中的内容。需要的画私聊我哈!!!
:::

一、简介

Java内存模型(Java Memory Model,简称JMM)是一种抽象的概念,它定义了Java程序中各个变量(包括实例域、静态域和数组元素)的读写访问方式,以及这些变量如何在多线程环境中被正确地同步和可见。JMM的目标是确保Java程序在各种不同的硬件和操作系统平台上都能保持一致的内存访问效果。

1、JMM的基本概念

  • 主内存与工作内存:在JMM中,所有的变量都存储在主内存中,而每个线程都有自己独立的工作内存(也称为本地内存或栈空间)。线程对变量的所有操作(读取、赋值等)都必须在自己的工作内存中进行,不能直接操作主内存中的变量。线程之间的变量值传递需要通过主内存来完成。
  • 内存间交互操作:JMM定义了8个操作来完成主内存和工作内存之间的交互,包括read(读取)、load(载入)、use(使用)、assign(赋值)、store(存储)、write(写入)、lock(锁定)和unlock(解锁)。这些操作必须按照一定的顺序执行,以确保线程之间的内存可见性和有序性。

2、JMM的关键特性

  • 原子性:JMM保证了基本数据类型的访问和赋值操作是原子的,但对于64位的数据类型(如long和double),在32位JVM中可能不具备原子性。此外,JMM允许通过synchronized和volatile关键字来保证复合操作的原子性。换句话说,一个操作是不可中断的。即使是在多个线程一起执行的时候,一个操作一旦开始,九不会被其他线程所干扰。
  • 可见性:可见性是指当一个线程修改了某个共享变量的值,其他线程能够立即感知到这个变化。JMM通过volatile关键字和synchronized关键字来保证变量的可见性。
  • 有序性:JMM允许编译器和处理器对指令进行重排序以提高性能,但重排序必须遵守数据的依赖性。通过volatile关键字可以禁止指令重排序,保证程序的有序性。

3、JMM的同步机制

  • synchronized:synchronized关键字可以用于方法或代码块上,它确保了同一时刻只有一个线程可以执行某个方法或代码块,从而实现了对共享变量的互斥访问。
  • volatile:volatile关键字用于修饰变量,它保证了变量的可见性和有序性(禁止指令重排序),但并不能保证原子性。

二、多线程基本操作

1、线程的五种状态

thread-status.png

  • NEW: 表示刚刚创建的线程,这个时候线程还没开始执行。
  • RUNABLE: 当调用 start() 方法时,处于该状态。表示线程所需的一切资源都已经准备好。
  • BLOCKED: 当线程在执行过程中遇到了锁,就会进入该状态。
  • WAITING: 处于无时间限制的等待状态。
  • TIME_WAITING:处于有限的等待状态。
  • TEERMINATED:当线程执行完毕,九进入结束状态。

2、interrup/isInterrupt/interrupted

线程中断不会使线程立即退出,而是给线程发送一个通知,告知目标线程,有人希望你退出了!至于目标线程接到通知后如何处理,则是由目标线程自行决定。
完全由目标线程自行决定这一点非常重要,如果正在运行业务过程,接收到通知你要停掉,就突然停掉会造成业务执行的不完整。
jdk 中关于线程中断提供了三个方法:

  • interrup(): 中断线程,添加中断状态。
  • isInterrupt(): 判断线程是否被中断。
  • interrupted(): 判断线程是否被中断,并清楚当前中断状态。

3、wait和notify

  • 这两个方法使 Object() 类提供的方法,也就是说任何对象都可以调用这两个方法。用于支持多线程之间的协作操作。
  • 线程 A 调用了 obj.wait() 方法,那么线程 A 就会停止继续运行,转为等待状态。那么等待何时才能结束呢?即:线程 A 会一直等到其他线程调用了 obj.notify() 方法为止。
  • 如果一个线程调用了 obj.wait() 那么它就会进入 obj 对象的等待队列。这个等待队列中,可能会有多个线程,因为系统运行多个线程同时等待某一个对象。当 obj.wait() 被调用时,它就会从这个等待队列中,随机选择一个线程,并将其唤醒。这种选择时不公平的,是完全随机的。

4、join 和 yield

当一个线程的输入非常依赖另一个或者多个线程的输出,此时,这个线程就需要等待依赖线程执行完毕,才能继续执行。JDK 提供了 join() 操作来实现这个功能。

三、volatile

正常情况下,如果我们不使用 volatile,那么每条线程都会有自己的缓存,当全局变量被修改时,其他线程可能并不会被通知到。

volatile 并不能真正的保证线程安全,它只能确保一个线程修改数据后,其他线程能看到这个改动。

目录
相关文章
|
9天前
|
监控 Java 调度
【Java学习】多线程&JUC万字超详解
本文详细介绍了多线程的概念和三种实现方式,还有一些常见的成员方法,CPU的调动方式,多线程的生命周期,还有线程安全问题,锁和死锁的概念,以及等待唤醒机制,阻塞队列,多线程的六种状态,线程池等
66 6
【Java学习】多线程&JUC万字超详解
|
2天前
|
Java 调度 开发者
Java并发编程:深入理解线程池
在Java的世界中,线程池是提升应用性能、实现高效并发处理的关键工具。本文将深入浅出地介绍线程池的核心概念、工作原理以及如何在实际应用中有效利用线程池来优化资源管理和任务调度。通过本文的学习,读者能够掌握线程池的基本使用技巧,并理解其背后的设计哲学。
|
2天前
|
缓存 监控 Java
Java中的并发编程:理解并应用线程池
在Java的并发编程中,线程池是提高应用程序性能的关键工具。本文将深入探讨如何有效利用线程池来管理资源、提升效率和简化代码结构。我们将从基础概念出发,逐步介绍线程池的配置、使用场景以及最佳实践,帮助开发者更好地掌握并发编程的核心技巧。
|
3天前
|
缓存 监控 Java
java中线程池的使用
java中线程池的使用
|
3天前
|
算法 Java 数据处理
Java并发编程:解锁多线程的力量
在Java的世界里,掌握并发编程是提升应用性能和响应能力的关键。本文将深入浅出地探讨如何利用Java的多线程特性来优化程序执行效率,从基础的线程创建到高级的并发工具类使用,带领读者一步步解锁Java并发编程的奥秘。你将学习到如何避免常见的并发陷阱,并实际应用这些知识来解决现实世界的问题。让我们一起开启高效编码的旅程吧!
|
8天前
|
存储 Java 程序员
优化Java多线程应用:是创建Thread对象直接调用start()方法?还是用个变量调用?
这篇文章探讨了Java中两种创建和启动线程的方法,并分析了它们的区别。作者建议直接调用 `Thread` 对象的 `start()` 方法,而非保持强引用,以避免内存泄漏、简化线程生命周期管理,并减少不必要的线程控制。文章详细解释了这种方法在使用 `ThreadLocal` 时的优势,并提供了代码示例。作者洛小豆,文章来源于稀土掘金。
|
5天前
|
Java 开发者
Java中的多线程编程基础与实战
【9月更文挑战第6天】本文将通过深入浅出的方式,带领读者了解并掌握Java中的多线程编程。我们将从基础概念出发,逐步深入到代码实践,最后探讨多线程在实际应用中的优势和注意事项。无论你是初学者还是有一定经验的开发者,这篇文章都能让你对Java多线程有更全面的认识。
14 1
|
11天前
|
Java 调度
Java中的多线程基础与实践
【8月更文挑战第31天】本文将深入浅出地讲解Java中多线程的基础知识,并通过实例展示如何在Java程序中实现多线程。我们将从多线程的基本概念出发,逐步深入到线程的创建、控制以及同步机制,最后通过一个简易版的生产者消费者模型来实践这些知识点。文章旨在帮助初学者快速掌握多线程编程的关键技能,并理解其背后的原理。
|
2天前
|
安全 Java UED
Java并发编程:解锁多线程的潜力
在Java的世界里,并发编程如同一场精心编排的交响乐,每个线程扮演着不同的乐手,共同奏响性能与效率的和声。本文将引导你走进Java并发编程的大门,探索如何在多核处理器上优雅地舞动多线程,从而提升应用的性能和响应性。我们将从基础概念出发,逐步深入到高级技巧,让你的代码在并行处理的海洋中乘风破浪。
|
11天前
|
传感器 C# 监控
硬件交互新体验:WPF与传感器的完美结合——从初始化串行端口到读取温度数据,一步步教你打造实时监控的智能应用
【8月更文挑战第31天】本文通过详细教程,指导Windows Presentation Foundation (WPF) 开发者如何读取并处理温度传感器数据,增强应用程序的功能性和用户体验。首先,通过`.NET Framework`的`Serial Port`类实现与传感器的串行通信;接着,创建WPF界面显示实时数据;最后,提供示例代码说明如何初始化串行端口及读取数据。无论哪种传感器,只要支持串行通信,均可采用类似方法集成到WPF应用中。适合希望掌握硬件交互技术的WPF开发者参考。
31 0