分析&疑问
第一个思考的问题是:线程池为什么关闭了?最新线程池、多线程系列面试题整理好了,点击Java面试库小程序在线刷题。
代码中并没有手动关闭的地方。看一下Executors.newSingleThreadExecotor
的源码实现:
这里创建的实际上是一个FinalizableDelegatedExecutorService
,这个包装类重写了finalize
函数,也就是说这个类会在被GC回收之前,先执行线程池的shutdown方法。
问题来了,GC只会回收不可达(unreachable)的对象,在submit
函数的栈帧未执行完出栈之前,executorService
应该是可达的才对。
更多多线程系列教程:https://www.javastack.cn/categories/Java/
对于此问题,先抛出结论:
当对象仍存在于作用域(stack frame)时,finalize
也可能会被执行
oracle jdk文档中有一段关于finalize的介绍:
也就是说,在jvm的优化下,可能会出现对象不可达之后被提前置空并回收的情况
从例子中可以看到,如果a在循环完成后已经不再使用了,则会出现先执行finalize的情况;虽然从对象作用域来说,方法没有执行完,栈帧并没有出栈,但是还是会被提前执行。
现在来增加一行代码,在最后一行打印对象a,让编译器/代码生成器认为后面有对象a的引用
... System.out.println(a); //打印结果 Created A@1be6f5c3 done. A@1be6f5c3
从结果上看,finalize方法都没有执行(因为main方法执行完成后进程直接结束了),更不会出现提前finalize的问题了。
基于上面的测试结果,再测试一种情况,在循环之前先将对象a置为null,并且在最后打印保持对象a的引用
从结果上看,手动置null的话也会导致对象被提前回收,虽然在最后还有引用,但此时引用的也是null了。
现在再回到上面的线程池问题,根据上面介绍的机制,在分析没有引用之后,对象会被提前finalize
可在上述代码中,return之前明明是有引用的executorService.execute(futureTask)
,为什么也会提前finalize呢?
猜测可能是由于在execute方法中,会调用threadPoolExecutor,会创建并启动一个新线程,这时会发生一次主动的线程切换,导致在活动线程中对象不可达
结合上面Oracle Jdk文档中的描述“可达对象(reachable object)是可以从任何活动线程的任何潜在的持续访问中的任何对象”,可以认为可能是因为一次显示的线程切换,对象被认为不可达了,导致线程池被提前finalize了。
下面来验证一下猜想:
执行若干时间后报错:
从错误上来看,“线程池”同样被提前shutdown了,那么一定是由于新建线程导致的吗?最新面试题整理好了,点击Java面试库小程序在线刷题。
下面将新建线程修改为Thread.sleep
测试一下:
执行结果一样是报错
由此可得,如果在执行的过程中,发生一次显式的线程切换,则会让编译器/代码生成器认为外层包装对象不可达
总结
虽然GC只会回收不可达GC ROOT的对象,但是在编译器(没有明确指出,也可能是JIT)/代码生成器的优化下,可能会出现对象提前置null,或者线程切换导致的“提前对象不可达”的情况。
所以如果想在finalize方法里做些事情的话,一定在最后显示的引用一下对象(toString/hashcode都可以),保持对象的可达性(reachable)
上面关于线程切换导致的对象不可达,没有官方文献的支持,只是个人一个测试结果,如有问题欢迎指出
综上所述,这种回收机制并不是JDK的bug,而算是一个优化策略,提前回收而已;但Executors.newSingleThreadExecutor
的实现里通过finalize来自动关闭线程池的做法是有Bug的,在经过优化后可能会导致线程池的提前shutdown,从而导致异常。
线程池的这个问题,在JDK的论坛里也是一个公开但未解决状态的问题:https://bugs.openjdk.java.net/browse/JDK-8145304。
不过在JDK11下,该问题已经被修复: