分享一道美团一面的面试题,简单又细腻

2024-02-25 16:38

本文主要是介绍分享一道美团一面的面试题,简单又细腻,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Photo By Instagram sooyaaa

问题 10

你用过 ThreadLocal 吗?它的实现原理是什么?虽然问题简洁明了直入主题,但是答案中却暗藏了很多坑。

我的答案

ThreadLocal 是一个可以提供线程本地变量的工具类,使用它声明的变量在每个线程访问的时候会单独的初始化变量的一个线程副本,这个副本与当前线程会关联起来,存储在线程本地达到与其他线程隔离的目的,从而提供线程本地变量的特性。这就是 ThreadLocal 类,想必每个 Java 程序员都应该知道这一点,所以在此就不多啰嗦了。说到 ThreadLocal,我们主要注意如下几个点:


它具体怎么存储的

ThreadLocal 会被存储在一个 ThreadLocalMap(ThreadLocal 的静态内部类) 的容器里面,当线程第一次访问一个 ThreadLocal 类型的变量的时候会初始化一个 ThreadLocalMap 实例作为当前线程的属性。然后将当前 ThreadLocal 作为 Key,值作为 Value 存储这个 map 当中。当该线程后续访问其他的 ThreadLocal 变量的时候就不需要重新初始化 map 了,所有 ThreadLocal 变量都会被存储在其中。


ThreadLocalMap 与 HashMap 有什么不同之处

说到 map 难免要和 HashMap 来对比一下,首先 ThreadLocalMap 专门是用来存储 ThreadLocal 的,所以设计比较简单。


元素包装

同样它内部存储是依赖于一个静态内部类 Entry 的数组,这个 Entry 的特殊之处在于它是一个 WeakReference,想必大家都知道 Java 中的四种引入类型:强,软,弱,虚。其中弱引用会被在垃圾回收时候直接回收掉,这也是为内存的垃圾回收埋下了伏笔。当 ThradLocal 的变量不会再被使用到的时候,下一次垃圾回收即可回收掉 ThreadLocalMap 中的 Entry 对象,而假设 Entry 不是 WeakReference,则只有等到线程销毁才会被回收掉,这样会导致一些垃圾数据长时间占用 JVM 的内存。


hash 冲突

我们都知道 HashMap 在发生冲突的时候会采用拉链法,将冲突的元素链式存储在同一个槽里面。而ThreadLocalMap 采取了另外一种方式,如果当前槽中已经有元素,那么它试图存入后一个槽中,直到找到可以容纳自己的槽。

会造成内存溢出吗

关于这一点,网上看到一大堆文章分析说 ThreadLocal 会发生内存泄漏问题,因为虽然被 WeakReference 的 Entry 被回收掉了,但是 Value 还在 ThreadLocalMap 中,只要线程一直存活着(例如线程池技术)就会始终保留着 Value。没错确实是这样,但是只要该线程再次操作 ThreadLocal 类型的变量,就会触发清理掉无用的 ThreadLocal 变量,这样就达到了垃圾清理的目的。所以我认为根本不需要考虑这种这种问题,除非你的 ThreadLocal 变量非常大,随便实例化几个就会把内存撑爆,如果真是这种情况,那么你需要考虑一下你的系统设计的合理性了。

如上即为我对 ThreadLocal 的理解,小伙伴是否认同呢?欢迎自行查看 ThreadLocal 源码来与我一起讨论。

以上即为昨天的问题的答案,小伙伴们对这个答案是否满意呢?欢迎留言和我讨论。

又要到年末了,你是不是又悄咪咪的开始看机会啦。为了广大小伙伴能充足电量,能顺利通过 BAT 的面试官无情三连炮,我特意推出大型刷题节目。每天一道题目,第二天给答案,前一天给小伙伴们独立思考的机会。

点下“在看”,鼓励一下?

这篇关于分享一道美团一面的面试题,简单又细腻的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/746061

相关文章

Golang操作DuckDB实战案例分享

《Golang操作DuckDB实战案例分享》DuckDB是一个嵌入式SQL数据库引擎,它与众所周知的SQLite非常相似,但它是为olap风格的工作负载设计的,DuckDB支持各种数据类型和SQL特性... 目录DuckDB的主要优点环境准备初始化表和数据查询单行或多行错误处理和事务完整代码最后总结Duck

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

使用IntelliJ IDEA创建简单的Java Web项目完整步骤

《使用IntelliJIDEA创建简单的JavaWeb项目完整步骤》:本文主要介绍如何使用IntelliJIDEA创建一个简单的JavaWeb项目,实现登录、注册和查看用户列表功能,使用Se... 目录前置准备项目功能实现步骤1. 创建项目2. 配置 Tomcat3. 项目文件结构4. 创建数据库和表5.

使用PyQt5编写一个简单的取色器

《使用PyQt5编写一个简单的取色器》:本文主要介绍PyQt5搭建的一个取色器,一共写了两款应用,一款使用快捷键捕获鼠标附近图像的RGB和16进制颜色编码,一款跟随鼠标刷新图像的RGB和16... 目录取色器1取色器2PyQt5搭建的一个取色器,一共写了两款应用,一款使用快捷键捕获鼠标附近图像的RGB和16

四种简单方法 轻松进入电脑主板 BIOS 或 UEFI 固件设置

《四种简单方法轻松进入电脑主板BIOS或UEFI固件设置》设置BIOS/UEFI是计算机维护和管理中的一项重要任务,它允许用户配置计算机的启动选项、硬件设置和其他关键参数,该怎么进入呢?下面... 随着计算机技术的发展,大多数主流 PC 和笔记本已经从传统 BIOS 转向了 UEFI 固件。很多时候,我们也

C#读取本地网络配置信息全攻略分享

《C#读取本地网络配置信息全攻略分享》在当今数字化时代,网络已深度融入我们生活与工作的方方面面,对于软件开发而言,掌握本地计算机的网络配置信息显得尤为关键,而在C#编程的世界里,我们又该如何巧妙地读取... 目录一、引言二、C# 读取本地网络配置信息的基础准备2.1 引入关键命名空间2.2 理解核心类与方法

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

Golang使用etcd构建分布式锁的示例分享

《Golang使用etcd构建分布式锁的示例分享》在本教程中,我们将学习如何使用Go和etcd构建分布式锁系统,分布式锁系统对于管理对分布式系统中共享资源的并发访问至关重要,它有助于维护一致性,防止竞... 目录引言环境准备新建Go项目实现加锁和解锁功能测试分布式锁重构实现失败重试总结引言我们将使用Go作

Python中列表的高级索引技巧分享

《Python中列表的高级索引技巧分享》列表是Python中最常用的数据结构之一,它允许你存储多个元素,并且可以通过索引来访问这些元素,本文将带你深入了解Python列表的高级索引技巧,希望对... 目录1.基本索引2.切片3.负数索引切片4.步长5.多维列表6.列表解析7.切片赋值8.删除元素9.反转列表