首页 > 其他分享 >五年前,我写错了一道面试题。

五年前,我写错了一道面试题。

时间:2024-04-22 12:44:23浏览次数:32  
标签:面试题 这个 抛出 五年 方法 写错 源码 线程 异常

你好呀,我是歪歪。

事情是这样的,上周有个读者找我,给我抛出了这样的一个问题:

问题中涉及到的文章分别是这两篇:

我自己写的这篇文章,虽然是五年前,2019 年的文章:

(卧槽,2019 年已经是五年前了)

但是毕竟是自己一个字一个字敲出来的,大概内容还是记得。

主要就是讨论了我在面试的时候遇到的这个问题:

一个线程池中的线程异常了,那么线程池会怎么处理这个线程?

当时我的回答是这样的:

在文章里面,我把我的回答总结成了三句话:

  • 1.抛出堆栈异常 ---这句话对了一半!
  • 2.不影响其他线程任务 ---这句话全对!
  • 3.这个线程会被放回线程池---这句话全错!

然后我的文章就基于上面这三句话展开了。

过程就不再赘述了,这次只讨论我五年前的文章中说错的一个点:这个(异常的)线程会被放回线程池。

当时我的结论是这句话全错了,正确的描述应该是:

(当一个线程池里面的线程异常后,)线程池会把这个线程移除掉,并创建一个新的线程放到线程池中。

对于同样的问题,京东技术的结论是这样的:

  • 当执行方式是 execute 时,可以看到堆栈异常的输出,线程池会把这个线程移除掉,并创建一个新的线程放到线程池中。
  • 当执行方式是 submit 时,堆栈异常没有输出。但是调用 Future.get() 方法时,可以捕获到异常,不会把这个线程移除掉,也不会创建新的线程放入到线程池中。

歪师傅的结论是一概而论,京东技术则是分情况讨论。

首先,京东技术的结论是正确的。

其次,歪师傅当年写这个文章的时候,就是技不如人,就是写错了,就是情况没有分析完整。

只看了 execute 的情况,导致得出了一个“只对了一半的答案”。

而关于使用 submit 方法时,如果在线程中抛出了异常,为什么不创建新的线程,而是继续复用原线程的原因,京东技术也从源码的角度解析了。

歪师傅这里也赘述一下。

问题的关键就是要抓到关键的问题。

那么在这个问题中,关键的问题是什么?

就是移除线程的方法在哪儿。

对应到源码其实就是这里:

java.util.concurrent.ThreadPoolExecutor#processWorkerExit

那么其实关键点就是这个方法在哪儿,在什么情况下会被调用到?

对应的源码在这里:

java.util.concurrent.ThreadPoolExecutor#runWorker

通过源码我们可以知道,在抛出异常的情况下,该方法会被调用到。

而 try 部分就只有一行代码:

task.run();

那么能耍花招的地方就只能是 task 这个对象了。

比如这样的代码,当 execute 方法执行的时候,这就是一个原生的 Thread 线程:

该方法是否会抛出异常,取决于你代码是否会抛出异常。

比如这样去写,线程执行 sayHi 方法的时候就会抛出异常:

而这样去写,则不会抛出异常:

所以,你再去看京东技术的结论:

execute 提交到线程池的方式,如果执行中抛出异常,并且没有在执行逻辑中 catch,那么会抛出异常,并且移除抛出异常的线程,创建新的线程放入到线程池中。

特别提到了 catch。

但是 submit 的时候,是怎么回事呢?

task 从一个普通线程变成了 FutureTask 对象:

因为源码在这里玩个了个小花招:

java.util.concurrent.AbstractExecutorService#submit(java.lang.Runnable)

把 task 包装成了 FutureTask 对象。

而一切的秘密就藏在 FutureTask 对象的 run 方法中:

java.util.concurrent.FutureTask#run

异常之后,会调用 setException 方法,仅仅是把异常放在了 outcome 字段中,然后维护了 FutureTask 的状态,不会继续往外抛出异常。

如果需要获取异常,则需要调用 get 方法。

好,现在我要开始闭环了。

因为 submit 提交的时候会把任务封装为 FutureTask 对象,该对象重写了 run 方法,所以当任务异常之后,不会继续往外抛出异常。

因为不会继续往外抛出异常,所以不会走到 processWorkerExit 方法。

因为不会走到 processWorkerExit 方法,所以不涉及移除线程和添加线程的逻辑。

所以:

当执行方式是 submit 时,不会把这个线程移除掉,也不会创建新的线程放入到线程池中。

其实整体逻辑还是很清楚的,当年就是分析漏了 submit 的情况,导致最终的结论不对。

五年前我挖了个坑,五年后,我把这个坑填一下。

然后再回答一个京东技术那篇文章下留言区的一个问题:

execute 执行无论是否抛出异常,finally 块中代码不是都会执行吗?

也就是这段代码:

如果你只看这部分 try 和 finally 代码块,我们学习 Java 的时候,如果老师没有骗我们的话,那么不管是正常执行完成 try 里面的代码,还是 try 里面的代码抛出异常, finally 代码块的代码理论上都是会执行的。

是的,这一个知识点没有任何毛病。

但是,你注意我是怎么说的“不管是正常执行完成还是抛出异常”。

抛出异常我们前面已经分析了,提问者的疑问点在于“正常执行完成”为什么不会执行 finally 代码块里面的 processWorkerExit 方法。

我的答案是:会。

但是,try 里面要正常执行完成,也就是 while 循环要正常结束,所以你看看一眼循环条件中的这个部分,要返回 null 才满足条件:

getTask 对应的源码是这样的:

java.util.concurrent.ThreadPoolExecutor#getTask

在我们讨论的场景下,线程是会阻塞在队列的 poll 或者 take 方法这里的。

如果是 take 方法就不说了,不会返回 null,在这里死等。

如果是 poll 方法返回了 null,则说明该线程到了超时时间还未从队列中获取到任务。

这个时候该怎么办?

翻翻八股文看看,如果线程池设置了 allowCoreThreadTimeOut 为 true,针对核心线程,在指定时间内未获取到任务或者非核心线程在指定时间内未获取到任务的时候,线程池会怎么处理?

是不是说的该销毁了,该从线程池中移走了?

所以,才会走到 processWorkerExit 执行 workers.remove(w) 方法。

是不是感觉自己又能行了,知识点又串起来了。

一点思考

当读者问我“是复用还是移除”这个问题的时候,我当时确实不知道答案。

但是我一点都不慌,因为我知道去哪里找答案。

如果我真的需要想要知道答案的话,在不借助任何搜索工具,仅仅给我源码的情况下,我应该很快就能得到一个准确的答案。

这一点自信的底气是因为我确实较为深入的研究过这部分源码。

但是当时我没有去寻找答案,结合我对于线程池的理解,我在思考另外一个问题:这重要吗?

你仔细想一想,如果这个问题抛出来之后你直接就是一头雾水,或者说和我一样知道去哪里找答案,那么这个问题的准确回答对你来说真的重要吗?

不管是那种情况都不重要,一点都不重要。

因为不管是销毁还是复用,它完全不影响你对于线程池的使用。

重要的是,在一头雾水的情况下,自己去寻找问题的答案的这个过程。

你当然可以拿着关键字去网上搜,肯定能搜到答案,这是一个寻找的过程,不过是轻松一点,然后遗忘起来快一点。

你也可以带着问题去翻源码,这也是一个寻找的过程,不过是难一点而已,记忆深刻一点。

如果觉得直接啃源码啃不动,那就结合网上的资料一起食用,这同样是一个寻找的过程。

等你真的找到这个问题的标准答案的时候、等你进一步理解线程池的时候,你会发现这个问题的答案不重要,但是在寻找的过程中你写的 Demo、接触到的源码、方法之间的调用关系、分支判断逻辑、查阅到的资料、付出的时间和对应的收获、甚至是内心中转瞬即逝的开心...

这些是重要的。

这个题其实是一个陷阱。

就像是我们读书的时候做的数学题,我们都知道参考答案就在练习册的最后几页,照着参考答案抄就能回答正确。

但是我们都知道比起正确答案来说,更重要的是你知道解题的过程。

最可怕的情况是你抄答案的次数多了,对自己产生了错误的认知,让你在抄答案的过程中还产生了这题很简单,自己也会做的错觉。

只有见过了无数千奇百怪的题目,摸熟了无数个解题的套路,当你在这个过程中,在某个瞬间体会到了“万变不离其宗”的时候,在自信心经历过建立、崩塌、再建立的过程后,在把参考答案真的只是当做参考的时候,你就可以淡定的说出:哦,这题啊,我没见过,但是我知道怎么去做。

就像是五年前我拿到这个题的时候,我经过一番研究,还是答错了。

五年后,再次遇到这个题的瞬间,我还是不知道答案,但是我的内心一点都不慌。

在学习编程的路上,这样的“陷阱题”真的太多太多了,难的不是回答出你被背下的标准答案,难的是你知道标准答案是怎么来的。

这就是我从“是复用还是移除”这个问题带给我的思考。

我觉得我其实是在试图给你阐述一种学习的方法,因为我也没有悟透,所以总感觉有点词不达意,但是我想要表述的都说完了,剩下的,我自己接着悟吧。

合订本

翻了一下,我过往还是写了很多线程相关的文章的。

都放在这里,作为一个合订版吧:

《有的线程它死了,于是它变成一道面试题》

《关于多线程中抛异常的这个面试题我再说最后一次!》

《如何设置线程池参数?美团给出了一个让面试官虎躯一震的回答。》

《填个坑!再谈线程池动态调整那点事。》

《每天都在用,但你知道 Tomcat 的线程池有多努力吗?》

《这个队列的思路真的好,现在它是我简历上的亮点了。》

《虽然是我遇到的一个棘手的生产问题,但是我写出来之后,就是你的了。》

《面试官:你给我说一下线程池里面的几把锁。》

《Dubbo 2.7.5在线程模型上的优化》

《面试官问我知不知道异步编程的Future。》

《面试官问我知不知道CompletionService?》

《1000 多个并发线程,10 台机器,每台机器 4 核,设计线程池大小。》

《要我说,多线程事务它必须就是个伪命题!》

《Doug Lea在J.U.C包里面写的BUG又被网友发现了。》

《“借助同步”这个理念在 FutureTask 里面的应用。》

《面试官:Java如何绑定线程到指定CPU上执行?》

《别问了,我真的不喜欢 @Asyn 这个注解!》

《看完JDK并发包源码的这个性能问题,我惊了!》

《什么是高并发下的请求合并?》

《CompletableFuture 的那点事儿》

《看起来是线程池的BUG,但是我认为是源码设计不合理。》

《喜提JDK的BUG一枚!多线程的情况下请谨慎使用这个类的stream遍历。》

《听我一句劝,业务代码中,别用多线程。》

《面试官:一个 SpringBoot 项目能处理多少请求?(小心有坑)》

《线程池参数千万不要这样设置》

《刺激,线程池的一个BUG直接把CPU干到100%了。》

《这里有线程池、局部变量、内部类、静态嵌套类和一个莫得名堂的引用,哦,还有一个坑!》

《看到一个魔改线程池,面试素材加一!》

《面试官一个线程池问题把我问懵逼了。》

如果里面的某一篇曾经帮助过你,安排一个一键三连就行了。

标签:面试题,这个,抛出,五年,方法,写错,源码,线程,异常
From: https://www.cnblogs.com/thisiswhy/p/18150414

相关文章

  • Java面试题:请谈谈对ThreadLocal的理解?
    ThreadLocal是一种特殊的变量存储机制,它提供了一种方式,可以在每个线程中保存数据,而不会受到其他线程的影响。这种机制在多线程编程中非常有用,因为它允许每个线程拥有自己的数据副本,从而避免了数据竞争和线程之间的干扰,以空间换时间。在Java中,ThreadLocal的实现主要涉及到三个类:Th......
  • 最新Java面试题带答案【2024中级】
    互联网大厂面试题1:阿里巴巴Java面试题2:阿里云Java面试题-实习生岗3:腾讯Java面试题-高级4:字节跳动Java面试题5:字节跳动Java面试题-大数据方向6:百度Java面试题7:蚂蚁金服Java面试题-中级8:蚂蚁金服Java面试题-高级9:京东Java面试题-中级10:拼多多Java面试题-电商部11:商汤科技......
  • Java面试题:为什么HashMap不建议使用对象作为Key?
    HashMap是一种基于哈希表的动态数据结构,它允许使用任意不可变对象作为键(key)来存储和检索数据。然而,在某些情况下,使用对象作为HashMap的键可能会遇到一些问题。 首先,我们需要明确对象作为HashMap的键需要满足一些条件:不可变性:对象的属性不能被修改,因为如果属性被修改,那......
  • 2024-04-19 前端常见面试题汇总(js篇)
    以下是前端面试中关于JavaScript的一些常见问题及其答案,共包含超过50个问题:1.解释一下JavaScript中的变量提升(Hoisting)。变量提升是指在JavaScript中,变量和函数的声明会被提升到其所在作用域的最顶部。但需要注意,只有声明会被提升,赋值操作不会。2.解释一下JavaScript中的闭包(C......
  • 2024-04-19 前端常见面试题汇总(html篇)
    1、xhtml和html有什么区别?语法要求:XHTML要求严格的XML语法,例如所有标签必须小写,所有标签必须关闭(即使是空元素也要使用闭合标签),所有属性必须使用引号。HTML语法相对更宽松,不强制要求标签闭合,标签和属性的大小写不敏感。文件类型:XHTML文档必须以.xml、.xhtml或者.xhtml......
  • 面试题:如何理解闭包
    之前看的闭包讲解,都是一些示例,不太好作为面试题作答内部函数如果引用了外部函数的变量,会形成闭包。如果这个内部函数作为外部函数的返回值,就会形成词法环境的引用闭环(循环应用),对应的变量就会常驻在内存中,形成大家所说的“闭包内存泄漏”。虽然闭包有内存上的问题,但是却突破了......
  • 前端面试题解析与总结
    在2024年的前端行业,面试是进入理想公司的一道门槛。不同公司的面试流程和考察点各有不同,下面将结合三家知名公司的面试题目进行分析和总结,为广大前端开发者提供一份参考指南。一、某对外电商一面:笔试题:弹窗组件防抖截流代码实现关系型数组转换成树形结构对象数组全排列......
  • 大数据面试题汇总
    大数据量场景面试题目录大数据量场景面试题假设有10亿手机号,如何快算判断一个手机号是否再其中?如何再海量数据中找到高频词?BitMap原理?BitMap应用?那么如何确定电话号码对应的是位图中的哪一位呢?假设有10亿手机号,如何快算判断一个手机号是否再其中?-无符号整数表示范围[0,1<......
  • 前端面试题 — webpack
    1.webpack的安装和使用方式安装Node.js和npm首先,确保你的计算机上安装了Node.js和npm(Node包管理器),因为Webpack是通过npm进行安装和管理的。创建项目目录并初始化npmnpminit-y安装Webpacknpminstallwebpackwebpack-cli......
  • 上海携程java高级面试题(一)
    一、JVM加载Class文件的原理机制?在面试java工程师的时候,这道题经常被问到,故需特别注意。Java中的所有类,都需要由类加载器装载到JVM中才能运行。类加载器本身也是一个类,而它的工作就是把class文件从硬盘读取到内存中。在写程序的时候,我们几乎不需要关心类的加载,因为这些都是隐式......