当前位置: 首页 > 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菜单栏的…...

区块链小故事

大灰狼与小白兔 一天兔子妈妈出门了&#xff0c;在大门上安装了一个区块链的门把手&#xff0c;这个门把手只有兔子妈妈、小兔子、以及另一个客人都同意的时候&#xff0c;才会开门&#xff0c;有一天客人a的钥匙丢了&#xff0c;被大灰狼捡到了&#xff0c;大灰狼于是去开门&…...

Java | Leetcode Java题解之第167题两数之和II-输入有序数组

题目&#xff1a; 题解&#xff1a; class Solution {public int[] twoSum(int[] numbers, int target) {int low 0, high numbers.length - 1;while (low < high) {int sum numbers[low] numbers[high];if (sum target) {return new int[]{low 1, high 1};} else i…...

项目训练营第三天

项目训练营第三天 注册登录测试 前面我们编写了用户注册、登录的逻辑代码&#xff0c;每编写完一个功能模块之后&#xff0c;我们都要对该模块进行单元测试&#xff0c;来确保该功能模块的正确性。一般情况下使用快捷键Ctrl Shift Insert&#xff0c;鼠标左击类名可以自动生…...

计算机组成原理 | CPU子系统(1)基本概述

基本结构模型 运算与缓存部件 数据寄存部件 PSW不是很清楚 存储器是什么&#xff1f;属于那个结构里&#xff1f; 时序处理部件 cpu是大脑&#xff0c;控制器是神经元 ①通过硬件产生控制信号 ②通过软件产生控制信号 外频&#xff08;系统时钟信号&#xff09;&#xff0c;…...

无引擎游戏开发(2):最简游戏框架 | EasyX制作井字棋小游戏I

一、EasyX中的坐标系 不同于数理中的坐标系&#xff0c;EasyX中的y轴是竖直向下的 二、渲染缓冲区 之前的程序添加了这三个函数改善了绘图时闪烁的情况: 小球在"画布“上移动的过程就是我们在调用绘图函数&#xff0c;这个”画布“就是渲染缓冲区&#xff0c;先绘制的内…...

排书 IDA*

原题链接 题目描述 给定 n 本书&#xff0c;编号为 1∼n。 在初始状态下&#xff0c;书是任意排列的。在每一次操作中&#xff0c;可以抽取其中连续的一段&#xff0c;再把这段插入到其他某个位置。我们的目标状态是把书按照 1∼n 的顺序依次排列。求最少需要多少次操作。 输…...

playwright录制脚本原理

Paywright录制工具UI 在上一篇博客中介绍了如何从0构建一款具备录制UI测试的小工具。此篇博客将从源码层面上梳理playwright录制原理。当打开playwright vscode插件时&#xff0c;点击录制按钮&#xff0c;会开启一个新浏览器&#xff0c;如下图所示&#xff0c;在新开浏览器页…...

awk脚本监控

awk脚本监控 使用脚本监控内存&#xff0c;cpu和硬盘的根目录&#xff0c;超过80%提示用户&#xff0c;写成函数库的行&#xff0c;每天早上 的8.50分&#xff0c;执行一次脚本 现在脚本中写需要的内容 cpuu () {aa$(top -b -n 1 |awk NR3 {printf "%.F",$2$4})if …...

Python高压电容导电体和水文椭圆微分

&#x1f3af;要点 &#x1f3af;二维热传导二阶偏微分方程 | &#x1f3af;调和函数和几何图曲率 | &#x1f3af;解潮汐波动方程 | &#x1f3af;解静止基态旋转球体流体运动函数 | &#x1f3af;水文空间插值 | &#x1f3af;流体流动模拟求解器 | &#x1f3af;随机算法解…...

微信小程序 引入MiniProgram Design失败

这tm MiniProgramDesign 是我用过最垃圾的框架没有之一 我按照官网的指示安装居然能安装不成功,牛! 这里说明我是用js开发的 到以上步骤没有报错什么都没有,然后在引入组件的时候报错 Component is not found in path “./miniprogram _npm/vant/weapp/button/index” (using…...