当前位置: 首页 > news >正文

java线程间的通信 - join 和 ThreadLocal

你好,我是 shengjk1,多年大厂经验,努力构建 通俗易懂的、好玩的编程语言教程。 欢迎关注!你会有如下收益:

  1. 了解大厂经验
  2. 拥有和大厂相匹配的技术等

希望看什么,评论或者私信告诉我!

文章目录

  • 一、前言
  • 二、join 和 ThreadLocal
    • 2.1 join 介绍
      • 2.1.1 join 作用和例子
      • 2.1.2 join 其他方法
      • 2.1.3 join 原理
      • 2.1.4 join 的使用场景
    • 2.2 ThreadLocal 介绍
      • 2.2.1 ThreadLocal 作用和例子
      • 2.2.2 ThreadLocal 原理
      • 2.2.3 ThreadLocal 的使用场景
      • 2.2.4 ThreadLocal 注意
  • 三、总结


一、前言

前面的几篇文章我们分别讲解了java多线程的基础知识、java 线程之间通信-volatile 和 synchronized 以及 java线程间的通信- notify和 wait本篇是java多线程通信的最后一篇:join 和 ThreadLocal

二、join 和 ThreadLocal

2.1 join 介绍

2.1.1 join 作用和例子

如果一个线程A执行了thread.join()语句,其含义是:当前线程A等待thread线程终止之后才从thread.join() 返回。
如:

public class ThreadJoin1 {public static void main(String[] args) throws InterruptedException {Thread thread = new Thread("threadB") {@Overridepublic void run() {try {TimeUnit.SECONDS.sleep(10);} catch (InterruptedException e) {e.printStackTrace();}}};thread.start();thread.join();System.out.println("threadB执行完成,主线程继续往下执行");}
}

这里主线程会一直等待threadB执行完成,才能继续往下执行

2.1.2 join 其他方法

线程Thread除了提供join()方法之外,还提供了join(long millis)和join(long millis,int nanos)两个具备超时特性的方法。这两个超时方法表示,如果线程thread在给定的超时时间里没有终止,那么将会从该超时方法中返回

2.1.3 join 原理

//当 join 的时间为 0 时
if (millis == 0) {while (isAlive()) {wait(0);}
} else {while (isAlive()) {long delay = millis - now;if (delay <= 0) {break;}wait(delay);now = System.currentTimeMillis() - base;}
}
  1. 等待线程终止join()方法是用于让当前线程等待调用该方法的线程执行完毕。主调用线程会等待被调用线程的终止。具体来说,调用join(millis)方法会使当前线程最多等待millis毫秒,直到被调用线程结束,设定超时为0表示永久等待,而不是等待当前线程自己终止。
  2. 实现机制join()方法的实现通常会使用this.wait()方法,顺序地调用this.wait(),并检查this.isAlive()条件,以等待被调用线程的终止。当被调用线程终止时,会调用notifyAll()方法来唤醒正在等待的线程,让其继续执行。
  3. 虽然源代码中无法找到join()方法内部的notifyAll()调用,但可以肯定的是,Java的多线程机制会确保在目标线程执行结束时,等待的线程能够被正确唤醒。这种机制是由Java运行时环境在底层实现的

2.1.4 join 的使用场景

join() 方法是用于让一个线程等待另一个线程完成的方法。在Java中,join() 方法通常用于以下应用场景:

  1. 等待线程完成后再执行后续操作:当一个线程需要等待另一个线程执行完成后才能继续执行,可以使用 join() 方法。这在需要线程间协作的情况下很有用。
  2. 多个线程按顺序执行:如果有多个线程按照一定的顺序来执行,可以使用 join() 方法使得后续的线程等待前一个线程完成之后再执行。
  3. 线程任务协调:当一个线程需要等待其他一组线程都完成后才能继续执行时,在这组线程中每个线程可以在执行完自己的任务后调用join()方法,主线程会等待所有这些线程执行完毕。
  4. 实现线程间的数据交换:通过join()方法,一个线程可以等待另一个线程计算出结果或者产生数据,从而实现多个线程间的数据交换。
  5. 任务分解:在复杂的任务分解中,join() 可以用来等待各个子任务完成,然后汇总结果。 举例来说,假设我们有一个主线程,需要等待两个子线程完成后再继续执行,可以这样实现:
public class JoinExample {public static void main(String[] args) {Thread thread1 = new Thread(() -> {// 执行一些任务System.out.println("Thread 1 finished");});Thread thread2 = new Thread(() -> {// 执行一些任务System.out.println("Thread 2 finished");});thread1.start();thread2.start();try {thread1.join(); // 主线程等待thread1完成thread2.join(); // 主线程等待thread2完成} catch (InterruptedException e) {e.printStackTrace();}// 所有线程都执行完毕后继续执行主线程System.out.println("All threads have finished");}
}

在这个例子中,主线程会等待thread1thread2 完成后才会继续执行。这种方式可以用于确保线程间的操作顺序性以及线程完成后的后续处理。

2.2 ThreadLocal 介绍

2.2.1 ThreadLocal 作用和例子

ThreadLocal,即线程变量( 可以理解为是线程的私有变量,每个线程一个所以不存在线程安全的问题 ),是一个以ThreadLocal对象为键、任意对象为值的存储结构。这个结构被附带在线程上,也就是说一个线程可以根据一个ThreadLocal对象查询到绑定在这个线程上的一个值。

public class MyThreadLocalExample {private static ThreadLocal<Integer> threadLocal = new ThreadLocal<>();public static void main(String[] args) {threadLocal.set(100);  // 在当前线程设置值System.out.println("Main Thread Value: " + threadLocal.get());Thread thread = new Thread(() -> {threadLocal.set(200);  // 在新线程设置值System.out.println("Child Thread Value: " + threadLocal.get());});thread.start();}
}

可以通过set(T)方法来设置一个值,在当前线程下再通过get()方法获取到原先设置的值

2.2.2 ThreadLocal 原理

ThreadLocal原理主要涉及Java内存模型中的“线程封闭”概念和ThreadLocal类的设计。

  1. 每个线程都有自己的ThreadLocalMap: 在Java中,每个线程都维护着一个ThreadLocalMap对象,其中存储了线程本地变量的键值对。这个ThreadLocalMap在Thread类中以ThreadLocal.ThreadLocalMap类型的成员变量存在。

  2. ThreadLocal作为 ThreadLocalMap 键: 当一个ThreadLocal对象被设置为线程的局部变量时,它实际上是作为ThreadLocalMap的键,而对应的值则是线程特定的变量。

  3. 支持多个ThreadLocal变量: 每个线程可以同时持有多个ThreadLocal对象,每个ThreadLocal对象对应一个特定的线程局部变量。

  4. ThreadLocalMap的查找和存储机制: 当线程需要获取ThreadLocal变量时,会先访问自己的ThreadLocalMap,根据ThreadLocal对象找到对应的值。如果没有找到,会创建一个新的Entry并存储到ThreadLocalMap中。

通过这种设计,ThreadLocal实现了线程封闭性和线程间变量隔离,为多线程环境下共享变量带来了一种简洁、安全的解决方案。

2.2.3 ThreadLocal 的使用场景

ThreadLocal 是一个线程本地变量工具类,它提供了线程级别的数据隔离,每个线程都可以拥有自己独立的变量副本。ThreadLocal 的使用场景包括但不限于:

  1. 保存线程相关的上下文信息:在多线程环境中,有些信息是针对每个线程独立存在的,比如用户身份信息、数据库连接、请求参数等。ThreadLocal 可以用于保存这些线程相关的上下文信息,避免在线程之间相互干扰。
  2. 提升性能:在某些情况下,对于频繁访问的对象,可以将其放入 ThreadLocal 中以减少方法参数传递和数据共享的开销,从而提升性能。
  3. 跟踪日志:可以利用 ThreadLocal 在日志中添加线程标识符,用于追踪不同线程产生的日志信息,方便日志分析和调试。
  4. 缓存数据:在某些情况下,需要对一些数据进行缓存,而这些数据是与线程相关的,可以使用 ThreadLocal 存储这些线程私有的缓存数据。
  5. Web开发中的用户会话管理:在 Web 开发中,可以使用 ThreadLocal 存储用户的会话信息,确保每个用户的会话在不同请求之间得到正确管理。

如,在一个多线程的 Web 应用中,需要持续跟踪用户的登录状态:

public class UserContext {private static ThreadLocal<User> userThreadLocal = new ThreadLocal<>();public static void setUser(User user) {userThreadLocal.set(user);}public static User getUser() {return userThreadLocal.get();}public static void clear() {userThreadLocal.remove();}
}// 在某个地方设置用户信息
User currentUser = userService.getCurrentUser();
UserContext.setUser(currentUser);// 在其他地方获取用户信息
User loggedInUser = UserContext.getUser();

2.2.4 ThreadLocal 注意

ThreadLocalMap 中的 entry 继承了 WeakReference,在内存不足的时候,ThreadLocalMap 中的数据会被自动回收,所以在使用的时候要注意处理空值情况

static class Entry extends WeakReference<ThreadLocal<?>> {/** The value associated with this ThreadLocal. */Object value;Entry(ThreadLocal<?> k, Object v) {super(k);value = v;}
}

三、总结

本文详细介绍了join方法的使用方法、原理和适用场景,强调了其在多线程编程中的重要性。同时,ThreadLocal的工作原理和实际应用场景也得到了阐述,特别是在数据隔离和性能优化方面的作用。

相关文章:

java线程间的通信 - join 和 ThreadLocal

你好&#xff0c;我是 shengjk1&#xff0c;多年大厂经验&#xff0c;努力构建 通俗易懂的、好玩的编程语言教程。 欢迎关注&#xff01;你会有如下收益&#xff1a; 了解大厂经验拥有和大厂相匹配的技术等 希望看什么&#xff0c;评论或者私信告诉我&#xff01; 文章目录 一…...

差分GPS原理

双差RTK&#xff08;Real-Time Kinematic&#xff09;算法是基于差分全球卫星导航系统&#xff08;GNSS&#xff09;技术的一种高精度定位方法。它利用至少两个接收机&#xff08;一个为基站&#xff0c;其他为移动站&#xff09;接收自同一组卫星的信号来实现精确测量。双差处…...

【栈与队列】前k个高频元素

题目&#xff1a;给你一个整数数组 nums 和一个整数 k &#xff0c;请你返回其中出现频率前 k 高的元素。你可以按 任意顺序 返回答案。 分析&#xff1a;首先我们需要计算数组中元素出现的频率&#xff0c;前几篇文章讲解了哈希表的应用&#xff0c;所以这里我们很容易想到用…...

B端产品竞品分析-总结版

B端竞品分析的难点 分析维度-业务逻辑复杂 B端产品与C端产品业务模型不同&#xff0c;B端产品主要以业务为导向&#xff0c;因此其业务流程与业务逻辑梳理起来也会较C端产品复杂的多&#xff0c;对于个人能力也有一定的要求&#xff0c;需要我们具备相关领域或行业专业知识。…...

刷代码随想录有感(116):动态规划——单词拆分

题干&#xff1a; 代码&#xff1a; class Solution { public:bool wordBreak(string s, vector<string>& wordDict) {unordered_set<string>set(wordDict.begin(), wordDict.end());vector<bool>dp(s.size() 1, false);dp[0] true;for(int j 0; j &…...

CSS-0_1 CSS和层叠(样式优先级、内联样式、选择器 用户代理样式)

CSS 的本质就是声明规则 ——《深入解析CSS》 文章目录 CSS层叠和优先级用户代理样式请和用户代理样式和谐相处 选择器单选择器的优先级选择器组的优先级关于选择器的其他源码顺序尽可能的选择优先级低的选择器 内联样式内联样式和JavaScript !important多个 !important 碎碎念…...

科技赋能冷链园区:可视化带来全新体验

应用图扑可视化技术&#xff0c;冷链园区能够更加直观地监控和管理资源&#xff0c;优化运作流程&#xff0c;提高运营效率与服务质量。...

高通安卓12-安卓系统定制2

将开机动画打包到system.img里面 在目录device->qcom下面 有lito和qssi两个文件夹 现在通过QSSI的方式创建开机动画&#xff0c;LITO方式是一样的 首先加入自己的开机动画&#xff0c;制作过程看前面的部分 打开qssi.mk文件&#xff0c;在文件的最后加入内容 PRODUCT_CO…...

高中数学:数列-解数列不等式问题的常用放缩技巧(重难点)

一、放缩技巧 技巧1 例题 证明&#xff1a;Sn&#xff1c;1 解&#xff1a; 变形 解&#xff1a; 由于第一种情况&#xff0c;我们证明了Sn&#xff1c;1&#xff0c;n≥1&#xff0c;是从第一项就开始放缩的。 发现&#xff0c;无法精确到 3 4 \frac{3}{4} 43​ 这时&am…...

[图解]企业应用架构模式2024新译本讲解17-活动记录1

1 00:00:01,070 --> 00:00:04,180 下一个我们要说的就是 2 00:00:04,190 --> 00:00:06,740 活动记录模式了 3 00:00:07,640 --> 00:00:11,210 同样是数据源架构模式 4 00:00:12,300 --> 00:00:18,480 里面的一个&#xff0c;活动记录 5 00:00:18,490 --> 00…...

[C++深入] --- malloc/free和new/delete

1 new运算符的拓展 1.1 自由存储区与堆的概念 在C++中,内存区分为5个区,分别是堆、栈、自由存储区、全局/静态存储区、常量存储区。 自由存储区是C++基于new操作符的一个抽象概念,凡是通过new操作符进行内存申请,该内存即为自由存储区。 new操作符从自由存储区(free st…...

Spcok测试代码抛异常场景

测试代码抛异常场景 ‍ class ExceptionSpec extends Specification {def validateService new ValidateService()Unrolldef "验证UserInfo"() {when: "调用校验方法"validateService.validateUser(user)then: "捕获异常并设置需要验证的异常值&qu…...

【漏洞复现】脸爱云一脸通智慧管理平台 SystemMng 管理用户信息泄露漏洞(XVE-2024-9382)

0x01 产品简介 脸爱云一脸通智慧管理平台是一套功能强大&#xff0c;运行稳定&#xff0c;操作简单方便&#xff0c;用户界面美观&#xff0c;轻松统计数据的一脸通系统。无需安装&#xff0c;只需在后台配置即可在浏览器登录。 功能包括:系统管理中心、人员信息管理中心、设备…...

新手如何入门Web3?

一、什么是Web3&#xff1f; Web3是指下一代互联网&#xff0c;它基于区块链技术&#xff0c;致力于将各种在线活动变得更加安全、透明和去中心化。Web3是一个广义的概念&#xff0c;涵盖了包括数字货币、去中心化应用、智能合约等在内的多个方面。它的主要特点包括去中心化、…...

React.FC`<ChildComponentProps>`解释

代码场景 ParentComponent.tsx import React, { useState } from react; import ChildComponent from ./ChildComponent;function ParentComponent() {const [childData, setChildData] useState<string>();const handleChildData (data: string) > { // 可以直接…...

2024-06-24力扣每日一题

链接&#xff1a; 503. 下一个更大元素 II 题意 循环数组&#xff0c;找出每个元素的往后最近且大于它的元素 解&#xff1a; 今天没试暴力啊&#xff0c;大概率是过不了的 思路就是先找到最大的数&#xff0c;最大数的结果肯定是-1&#xff0c;然后倒着遍历数组&#xf…...

pyhon模块以及常用的第三方模块

import my_info as info print(info.name) info.show()from my_info import * print(name) show() pyhon中包的导入 import admin.my_admin as ad # 包名.模块名 admin是包名&#xff0c;my_admin是模块名print(ad.name) print(ad.info())from admin import my_admin as ad # …...

shell脚本—快速修改centos网络配置

shell-文本中自行修改想要的配置 #!/bin/bash# 网卡名称 eth"eth0"# IP 地址 ipaddr"192.168.1.100"# 子网掩码 netmask"255.255.255.0"# 网关 gateway"192.168.1.1"# 写入配置文件 echo "BOOTPROTOstatic" > /etc/sysc…...

线程池概念、线程池的不同创建方式、线程池的拒绝策略

文章目录 &#x1f490;线程池概念以及什么是工厂模式&#x1f490;标准库中的线程池&#x1f490;什么是工厂模式&#xff1f;&#x1f490;ThreadPoolExecutor&#x1f490;模拟实现线程池 &#x1f490;线程池概念以及什么是工厂模式 线程的诞生是因为&#xff0c;频繁的创…...

示例:WPF中如何绑定ContextMenu和Menu

一、目的&#xff1a;开发过程中&#xff0c;有些模块的右键ContextMenu菜单是需要动态显示的&#xff0c;既是根据不同条件显示不同的菜单&#xff0c;很多是通过代码去生成ContextMenu的MenuItem&#xff0c;本文介绍通过绑定的方式去加载ContextMenu&#xff0c;Menu菜单栏的…...

VB.net复制Ntag213卡写入UID

本示例使用的发卡器&#xff1a;https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...

循环冗余码校验CRC码 算法步骤+详细实例计算

通信过程&#xff1a;&#xff08;白话解释&#xff09; 我们将原始待发送的消息称为 M M M&#xff0c;依据发送接收消息双方约定的生成多项式 G ( x ) G(x) G(x)&#xff08;意思就是 G &#xff08; x ) G&#xff08;x) G&#xff08;x) 是已知的&#xff09;&#xff0…...

2021-03-15 iview一些问题

1.iview 在使用tree组件时&#xff0c;发现没有set类的方法&#xff0c;只有get&#xff0c;那么要改变tree值&#xff0c;只能遍历treeData&#xff0c;递归修改treeData的checked&#xff0c;发现无法更改&#xff0c;原因在于check模式下&#xff0c;子元素的勾选状态跟父节…...

Python爬虫(二):爬虫完整流程

爬虫完整流程详解&#xff08;7大核心步骤实战技巧&#xff09; 一、爬虫完整工作流程 以下是爬虫开发的完整流程&#xff0c;我将结合具体技术点和实战经验展开说明&#xff1a; 1. 目标分析与前期准备 网站技术分析&#xff1a; 使用浏览器开发者工具&#xff08;F12&…...

在WSL2的Ubuntu镜像中安装Docker

Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包&#xff1a; for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...

Spring AI与Spring Modulith核心技术解析

Spring AI核心架构解析 Spring AI&#xff08;https://spring.io/projects/spring-ai&#xff09;作为Spring生态中的AI集成框架&#xff0c;其核心设计理念是通过模块化架构降低AI应用的开发复杂度。与Python生态中的LangChain/LlamaIndex等工具类似&#xff0c;但特别为多语…...

Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习)

Aspose.PDF 限制绕过方案&#xff1a;Java 字节码技术实战分享&#xff08;仅供学习&#xff09; 一、Aspose.PDF 简介二、说明&#xff08;⚠️仅供学习与研究使用&#xff09;三、技术流程总览四、准备工作1. 下载 Jar 包2. Maven 项目依赖配置 五、字节码修改实现代码&#…...

基于 TAPD 进行项目管理

起因 自己写了个小工具&#xff0c;仓库用的Github。之前在用markdown进行需求管理&#xff0c;现在随着功能的增加&#xff0c;感觉有点难以管理了&#xff0c;所以用TAPD这个工具进行需求、Bug管理。 操作流程 注册 TAPD&#xff0c;需要提供一个企业名新建一个项目&#…...

排序算法总结(C++)

目录 一、稳定性二、排序算法选择、冒泡、插入排序归并排序随机快速排序堆排序基数排序计数排序 三、总结 一、稳定性 排序算法的稳定性是指&#xff1a;同样大小的样本 **&#xff08;同样大小的数据&#xff09;**在排序之后不会改变原始的相对次序。 稳定性对基础类型对象…...

API网关Kong的鉴权与限流:高并发场景下的核心实践

&#x1f525;「炎码工坊」技术弹药已装填&#xff01; 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 引言 在微服务架构中&#xff0c;API网关承担着流量调度、安全防护和协议转换的核心职责。作为云原生时代的代表性网关&#xff0c;Kong凭借其插件化架构…...