生成和管理后台线程的惯用 Clojure 方式
创建一个在后台循环更新某些共享引用并管理其生命周期的线程的惯用 Clojure 方法是什么?我发现自己使用 future
来实现此目的,但这感觉有点像黑客,因为我从未返回有意义的值。例如:
(future (loop [] (do
(Thread/sleep 100)
(dosync (...))
(recur))))
另外,当不再需要后台处理时,我需要小心地future-cancel
。任何关于如何在 Clojure/Swing 应用程序中编排它的提示都会很好。例如,添加到我的 UI 中的虚拟 JComponent
负责在窗口关闭时杀死线程可能是一个想法。
What is the idiomatic Clojure way to create a thread that loops in the background doing updates to some shared refs and to manage its lifetime? I find myself using future
for this, but it feels like a little bit of a hack as I never return a meaningful value. E.g.:
(future (loop [] (do
(Thread/sleep 100)
(dosync (...))
(recur))))
Also, I need to be careful to future-cancel
this when the background processing is no longer needed. Any tips on how to orchestrate that in a Clojure/Swing application would be nice. E.g. a dummy JComponent
that is added to my UI that is responsible for killing the thread when the window is closed may be an idea.
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
您的循环中不需要
do
;这是暗示的。另外,虽然无条件循环递归没有任何问题,但您也可以使用 (while true ...)。future
是一个很好的工具;不要因为永远得不到回报而烦恼。不过,如果您使用代理而不是未来,那么这确实会困扰您 - 没有价值观的代理是疯狂的。然而,谁说你需要
future-cancel
?只需将您未来的步骤之一作为检查是否仍然需要它。然后代码的其他部分不需要跟踪 future 并决定何时取消它们。所以类似的方法将是一种可行的方法。
You don't need a
do
in your loop; it's implied. Also, while there's nothing wrong with an unconditional loop-recur, you may as well use (while true ...).future
is a fine tool for this; don't let it bother you that you never get a value back. That should really bother you if you use an agent rather than a future, though - agents without values are madness.However, who said you need to
future-cancel
? Just make one of the steps in your future be to check whether it's still needed. Then no other parts of your code need to keep track of futures and decide when to cancel them. So something likewould be a viable approach.
使用代理执行后台重复任务对我来说似乎更简洁
现在您所要做的就是启动循环
my-backgound-task
函数在调用完成后将其自身发送到调用代理。这是 Rich Hickey 在蚁群示例应用程序中执行重复任务的方式:Clojure Concurrency
Using agents for background recurring tasks seems neater to me
Now all you have to do is to initiate the loop
The
my-backgound-task
function is sending itself to the calling agent after its invocation is done.This is the way how Rich Hickey performs recurring tasks in the ant colony example application: Clojure Concurrency