作者热门文章
- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有以下代码:
object KafkaApi {
private implicit val main: ExecutionContextExecutor = ExecutionContext.global
private val workers = ExecutionContext.fromExecutor(Executors.newCachedThreadPool())
def main(args: Array[String]) {
foo.unsafeRunAsync(_ => ())
//foo.unsafeRunSync()
println("Hello")
}
def foo: IO[Unit] =
for {
_ <- IO {
println(Thread.currentThread().getName)
}
_ <- IO.shift(workers)
_ <- IO {
println(Thread.currentThread().getName)
}
_ <- IO {
println(Thread.currentThread().getName)
}
_ <- IO {
println(Thread.currentThread().getName)
}
_ <- IO.shift(main)
_ <- IO {
println(Thread.currentThread().getName)
}
_ <- IO {
println(Thread.currentThread().getName)
}
_ <- IO {
println(Thread.currentThread().getName)
}
} yield ()
}
main
Hello
pool-1-thread-1
pool-1-thread-1
pool-1-thread-1
scala-execution-context-global-14
scala-execution-context-global-14
scala-execution-context-global-14
最佳答案
Running the code above, why the application never get terminated?
Thread
都分为“普通”和
"daemon"线程。这里重要的是
The Java Virtual Machine exits when the only threads running are all daemon threads.
Thread
,则JVM会认为您的应用程序仍在工作,即使它实际上什么也不做(也许只是在等待一些输入)。 “主”线程显然是“普通”线程。由标准
ExecutionContext.global
创建的线程是守护程序,因此不会在主线程完成后阻止您的应用程序退出。由Java的
Executors.newCachedThreadPool
创建的线程是非守护程序,因此可以使应用程序保持 Activity 状态。有几种可能的解决方案:
ExecutionContext
外,请勿使用其他global
,即完全不使用Executors.newCachedThreadPool
。根据您的情况,这可能是您想要的,也可能不是。 shutdown
您的自定义ExecutorService
。这里要小心,因为shutdown
不会等待所有 Activity 任务完成。所以代码应该变成像private val pool = Executors.newCachedThreadPool
implicit private val workers = ExecutionContext.fromExecutor(pool)
// do whatever you want with workers
// optionally wait for all the work to be done
pool.shutdown()
val workers = ExecutionContext.fromExecutor(Executors.newCachedThreadPool(new ThreadFactory {
private val defaultDelegate = Executors.defaultThreadFactory()
override def newThread(r: Runnable): Thread = {
val t = defaultDelegate.newThread(r)
//let the default factory do all the job and just override daemon-flag
t.setDaemon(true)
t
}
}))
shutdown
,这很方便。代价是,如果由于某种原因您在执行其他所有任务之前判断错误并退出了“主”线程,您将不会知道任何错误,因为守护程序线程将被静默杀死。如果您使用#2并发生相同的错误,或者您未在该代码路径中调用
shutdown
,则您的应用将继续运行,或者您会在日志中看到一些警告,指出在有一些任务的情况下池已关闭进行中。因此,如果这只是一长串处理的中间步骤,出于某种原因需要自定义线程池,那么我可能会选择#3;但是如果这种并行执行是主要行为,那么我将采用更明确的#2方法。
关于multithreading - ExecutionContext.global和主线程之间的区别,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47578313/
有人可以向我澄清主线 DHT 规范中的声明吗? Upon inserting the first node into its routing table and when starting up th
我正在尝试使用 USB 小工具驱动程序使嵌入式设备作为 MTP 设备工作。 我知道 Android 从大容量存储设备切换到 MTP 设备已经有一段时间了,并且找到了 source code for M
我是一名优秀的程序员,十分优秀!