Looper如何通过ThreadLocal保证的线程独有

ThreadLocalMap

每个线程Thread都会维护一个threadLocals变量:ThreadLocalMap。这个类是ThreadLocal的静态内部类。当进行ThreadLocal的set()和get()时都会去获取线程里面的这个threadLocals进行设值和取值操作。所以ThreadLocal可以做到各个线程间数据互不干扰

ThreadLocal的set()方法及扩容机制

当调用ThreadLocal的set()方法时会获取当前线程的ThreadLocalMap。如果ThreadLocalMap不为空直接调用set()方法,key为thisThreadLocal,value为数据;如果ThreadLocalMap为空那么就会创建一个ThreadLocalMap并把数据添加到map中

ThreadLocalMap中的set()方法会首先进行查找存放这个Entry的数组。下标用key的hash值和数组长度-1进行与运算(key.threadLocalHashCode & (len-1)),之后在数组中从该下标位置向后遍历找位置如果找到的Enrtykey值相同则覆盖值并退出方法,如果找到的Entrykey值为null则在该位置把数据添加进来并退出方法;如果该下标后面既没有key值相同的也没有key值null的,那么才会在该下标处进行添加Entry,这种情况下要再次进行对该下标后面的Entry清理观察是否有条目删除。如果后面的条目都是有用的没有被删除并且使用的数组超过了数组长度的三分之二,那么会进行一次rehash的调用该方法会对整个数组进行一次检查并且删除无用的Entry(key为null),之后会判断是否超出了阈值0.75如果超出进行扩容,数组长度直接扩大一倍

ThreadLocal中有很多和Hashmap类似的计算。比如数组下标都是对key的hash值和数组长度-1进行与运算。阈值也为数组长度的0.75如果超出则进行扩容并且都是扩一倍

注意:找到下标后并不是直接添加到数组中的该下标下面,而是先用直接寻址法去寻找该下标后面的Entry中是否有存放过或者后面的Enrty是已经无用的,找到后直接赋值上去或者覆盖值。如果没有找到才在该下标下面存放,存放完要判断一下是否超出了数组长度的三分之二,超出的话进行数组里面所有元素一次大面积的检查和清空(检查key为null)并且会判断一下是否超出阈值,如果清理完还是超出了阈值0.75那么会进行数组长度的扩容(原来的一倍)

ThreadLocal导致的内存泄漏

因为ThreadLocal创建一般是在线程里面或者Activity中创建,所以默认ThreadLocal会拿着线程或者Activity的引用,而threadLocalMap又会拿着Threadlocal的引用,如果没有对ThreadLocalMap里面的Entry进行删除,那么线程销毁或者Activity销毁的时候就会导致泄漏。

在Map内部是继承的弱引用保存key是拿的弱引用保存这其实是ThreadLocal内部的一种保护机制,还有在set和get的时候都会对key为null的Entry进行释放或者重新利用。但是我们还是需要对于无用数据进行手动调用remove,而不是借助JVM的被动GC

、 还有一种情况 ThreadLocalMap是线程内部的所以除了上面那种防止拿着外部引用导致的泄漏,还要防止线程Thread内部的泄漏。解决方法:自己创建的线程在销毁的时候要对ThreadLocalMap内部的数据进行remove

ThreadLocal的get方法

当调用ThreadLocal的get()方法时会获取当前线程的ThreadLocalMap。如果ThreadLocalMap为空那么会进行赋默认值,在赋默认值的时候还会尝试去取ThreadLocalMap如果获取到的还是空那么会初始化map并且赋默认值,这个默认值可以自己重写指定,添加到数组中之后返回这个默认值;如果获取到了map那么会获取值Entry然后拿到value进行返回

获取值也是根据下标去寻找(key.threadLocalHashCode & (len-1))。在Map内部这样实现:首先根据这个下标位置找,如果找到直接返回这个Entry获取值返回,如果这个下标没找到那么会**在这个下标后面查找找到无用的条目进行清除,找到符合条件的key也会直接返回,如果沒有找到那么返回null

原文链接:Looper如何通过ThreadLocal保证的线程独有 - 掘金 (juejin.cn)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/605440.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python 写自动点击爬取数据

今天来点不一样的!哥们 提示: 这里只是用于自己学习的 ,请勿用违法地方 效果图 会进行点击下一页 进行抓取 需要其他操作也可以自己写 文章目录 今天来点不一样的!哥们前言一、上代码?总结 前言 爬虫是指通过编程自动…

Spark基础原理

Spark On Yarn Spark On Yarn的本质 Spark专注于分布式计算,Yarn专注于资源管理,Spark将资源管理的工作交给了Yarn来负责 Spark On Yarn两种部署方式 Spark中有两种部署方式,Client和Cluster方式,默认是Client方式。这两种方式的本质区别&#xff0c…

[C#]winform利用seetaface6实现C#人脸检测活体检测口罩检测年龄预测性别判断眼睛状态检测

【官方框架地址】 https://github.com/ViewFaceCore/ViewFaceCore 【算法介绍】 SeetaFace6是由中国科技公司自主研发的一款人脸识别技术,它基于深度学习算法,能够快速、准确地识别出人脸,并且支持多种应用场景,如门禁系统、移动…

智慧校园电子班牌管理系统源码 Java Android原生

智慧校园电子班牌系统源码是一种智能化的教育管理解决方案,它可以在学校内实现信息共享、教学管理、学生管理、家校互通等各个方面的协调与配合,帮助教师、学生和家长更加高效地开展教学活动、管理学生、协同合作,从而推动学校教育水平的提高…

【设计模式之美】 SOLID 原则之四:接口隔离原则有哪三种应用?原则中的“接口”该如何理解?

文章目录 一. 如何理解“接口隔离原则”?二. 接口隔离的三个情况1. 把“接口”理解为一组 API 接口集合2. 把“接口”理解为单个 API 接口或函数 ing(能够使用的场景并未理解)3. 把“接口”理解为 OOP 中的接口概念 一. 如何理解“接口隔离原…

Redis内存策略:「过期Key删除策略」+ 「内存淘汰策略」

Redis之所以性能强,最主要的原因就是基于内存存储,然而单节点的Redis其内存大小不宜过大,否则会影响持久化或主从同步的性能。 Redis内存满了,会发生什么? 在Redis的运行内存达到了某个阈值,就会触发内存…

图神经网络入门

图神经网络(GNN)是一组在图领域工作的深度学习方法。 这些网络最近已应用于多个领域,包括: 组合优化、推荐系统、计算机视觉—仅举几例。 这些网络还可用于对大型系统进行建模,例如社交网络、蛋白质-蛋白质相互作用网络…

Python 数据类型有哪些?

Python 数据类型 Python 数据类型用于定义变量的类型。它定义了我们将在变量中存储哪种类型的数据。存储在内存中的数据可以有很多种类型。例如,一个人的年龄被存储为数值,他或她的地址被存储为字母数字字符。 Python 有各种内置数据类型,主…

软件测试方法分类-按测试对象划分

接上一篇,下来我们再细讲,第四个维度的分类, 软件测试方法分类-按测试对象划分 本章节重点介绍非功能测试的相关知识,因为功能测试的基本在之前的分类都是有涉及的。 一、非功能测试 1,性能测试(Performance Testing) 检查系统是否满足需求规格说明书中规定的性能。 …

pytorch07:损失函数与优化器

目录 一、损失函数是什么二、常见的损失函数2.1 nn.CrossEntropyLoss交叉熵损失函数2.1.1 交叉熵的概念2.2.2 交叉熵代码实现2.2.3 加权重损失 2.2 nn.NLLLoss2.2.1 代码实现 2.3 nn.BCELoss2.3.1 代码实现 2.4 nn.BCEWithLogitsLoss2.4.1 代码实现 三、优化器Optimizer3.1 什么…

android常用方法

获取应用安装来源 private String getAppInstaller(Context context, String packageName) {return context.getPackageManager().getInstallerPackageName(packageName);}判断是否系统应用 在/system/app 或者 /system/priv-app目录下的应用。 public boolean isSystem(Conte…

Redis入门-五种数据类型

大家好我是苏麟 , 今天来说说Reids五种基本数据类型 . 五种常用数据类型介绍 大纲 字符串操作命令哈希操作命令列表操作命令集合操作命令有序集合操作命令通用命令 Redis存储的是key-value结构的数据,其中key是字符串类型,value有5种常用的数据类型&…

导入导出(excel格式/压缩包格式导出至response、excel导入)

目录 导出一、excel格式导出至response1、List<对象>导出2、json对象不固定列导出 二、压缩包格式导出至response使用Hutool工具简化代码 导入一、对象导入二、不固定列导入 导出 一、excel格式导出至response 1、List<对象>导出 Map<String, String> hea…

零基础学Python网络爬虫案例实战全流程详解(入门与提高篇)

零基础学Python网络爬虫案例实战 全流程详解 入门与提高篇 零基础学Python网络爬虫案例实战 全流程详解 高级进阶篇 内容简介 在大数据时代的今天&#xff0c;无论是产品开始还是精准化营销越来越离不开大数据的支持&#xff0c;如何从浩瀚的网络中获取自己想要的大数据&…

用 Socket.D 替代原生 WebSocket 做前端开发

socket.d.js 是基于 websocket 包装的 socket.d 协议的实现。就是用 ws 传输数据&#xff0c;但功能更强大。 功能原生 websocketsocket.d说明listen有有监听消息send有有发消息sendAndRequest无有发消息并接收一个响应&#xff08;类似于 http&#xff09;sendAndSubscribe无…

PyTorch 简单易懂的实现 CosineSimilarity 和 PairwiseDistance - 距离度量的操作

目录 torch.nn子模块Distance Functions解析 nn.CosineSimilarity 功能 主要参数 输入和输出的形状 使用示例 nn.PairwiseDistance 功能 主要参数 输入和输出的形状 使用示例 总结 torch.nn子模块​​​​​​​Distance Functions解析 nn.CosineSimilarity torc…

【图神经网络导论】之第9章模型变体(刘知远)

第9章不同图类型的模型变体 文章目录 第9章不同图类型的模型变体9.1 有向图9.2 异构图9.3 带有边信息的图9.4 动态图9.5 多维图 第4章介绍的基础GNN模型"被用于处理无向图&#xff0c;这些图包含具有标签的节点&#xff0c;是最简单的图。然而&#xff0c;在现实世界中还有…

Linux基础知识点(八-POSXI互斥锁)

目录 一、互斥锁基本概念 1.1 互斥相关背景概念 1.2 互斥锁(mutex) 1.3 死锁 二、初始化互斥锁 2.1 静态初始化 2.2 动态初始化 三、获取与释放互斥锁 四、销毁互斥锁 一、互斥锁基本概念 1.1 互斥相关背景概念 临界资源&#xff1a;多线程执行流共享的资源就叫做…

【事务】事务传播级别

Spring事务定义了7种传播机制&#xff1a; PROPAGATION_REQUIRED&#xff1a;默认的Spring事物传播级别&#xff0c;若当前存在事务&#xff0c;则加入该事务&#xff0c;若不存在事务&#xff0c;则新建一个事务。 PAOPAGATION_REQUIRE_NEW&#xff1a;若当前没有事务&#x…

PyTorch 简单易懂的 Embedding 和 EmbeddingBag - 解析与实践

目录 torch.nn子模块Sparse Layers详解 nn.Embedding 用途 主要参数 注意事项 使用示例 从预训练权重创建嵌入 nn.EmbeddingBag 功能和用途 主要参数 使用示例 从预训练权重创建 总结 torch.nn子模块Sparse Layers详解 nn.Embedding torch.nn.Embedding 是 PyTo…