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

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

相关文章

使用Python开发一个简单的本地图片服务器

《使用Python开发一个简单的本地图片服务器》本文介绍了如何结合wxPython构建的图形用户界面GUI和Python内建的Web服务器功能,在本地网络中搭建一个私人的,即开即用的网页相册,文中的示... 目录项目目标核心技术栈代码深度解析完整代码工作流程主要功能与优势潜在改进与思考运行结果总结你是否曾经

Mysql表的简单操作(基本技能)

《Mysql表的简单操作(基本技能)》在数据库中,表的操作主要包括表的创建、查看、修改、删除等,了解如何操作这些表是数据库管理和开发的基本技能,本文给大家介绍Mysql表的简单操作,感兴趣的朋友一起看... 目录3.1 创建表 3.2 查看表结构3.3 修改表3.4 实践案例:修改表在数据库中,表的操作主要

springboot简单集成Security配置的教程

《springboot简单集成Security配置的教程》:本文主要介绍springboot简单集成Security配置的教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录集成Security安全框架引入依赖编写配置类WebSecurityConfig(自定义资源权限规则

如何使用Python实现一个简单的window任务管理器

《如何使用Python实现一个简单的window任务管理器》这篇文章主要为大家详细介绍了如何使用Python实现一个简单的window任务管理器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 任务管理器效果图完整代码import tkinter as tkfrom tkinter i

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

Python解析器安装指南分享(Mac/Windows/Linux)

《Python解析器安装指南分享(Mac/Windows/Linux)》:本文主要介绍Python解析器安装指南(Mac/Windows/Linux),具有很好的参考价值,希望对大家有所帮助,如有... 目NMNkN录1js. 安装包下载1.1 python 下载官网2.核心安装方式3. MACOS 系统安

使用EasyExcel实现简单的Excel表格解析操作

《使用EasyExcel实现简单的Excel表格解析操作》:本文主要介绍如何使用EasyExcel完成简单的表格解析操作,同时实现了大量数据情况下数据的分次批量入库,并记录每条数据入库的状态,感兴... 目录前言固定模板及表数据格式的解析实现Excel模板内容对应的实体类实现AnalysisEventLis

Java中数组转换为列表的两种实现方式(超简单)

《Java中数组转换为列表的两种实现方式(超简单)》本文介绍了在Java中将数组转换为列表的两种常见方法使用Arrays.asList和Java8的StreamAPI,Arrays.asList方法简... 目录1. 使用Java Collections框架(Arrays.asList)1.1 示例代码1.

Java嵌套for循环优化方案分享

《Java嵌套for循环优化方案分享》介绍了Java中嵌套for循环的优化方法,包括减少循环次数、合并循环、使用更高效的数据结构、并行处理、预处理和缓存、算法优化、尽量减少对象创建以及本地变量优化,通... 目录Java 嵌套 for 循环优化方案1. 减少循环次数2. 合并循环3. 使用更高效的数据结构4

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链