我有一个特定的线程,当 CF9 应用程序启动时我会启动该线程。它管理一个队列并从队列中取出项目来处理它们。如果没有可以从队列中取出的项目,它将sleep()
。该线程可以存活数周,进行处理和 hibernate 等。
我遇到的问题是,我从队列中取出的项目在我使用完它们很久之后仍保留在堆中(看起来像在老一代中)。
JVM 大约每小时完成一次完整的垃圾收集(我可以从 jConsole 看出这一点),即使运行时,我处理过的项目仍然保留在堆中。我之所以能这么说,是因为我执行了 jmap
堆转储并使用内存分析工具 Eclipse 插件对其进行了分析。
这是我的代码,在 Application.cfc
中执行:
<!--- Kick off a new thread and run through the queue --->
<cfthread action="run" name="myThread">
<cfloop condition="APPLICATION.threadProcessing eq true">
<cfif APPLICATION.myQueue.hasNext()>
<!--- Get next item --->
<cfset tmpItem = APPLICATION.myQueue.next() />
<!--- Ask item to process itself --->
<cfset tmpItem.process() />
<!--- PROBLEM: these 'tmpItem' objects are never cleaned up by the garbage collector! --->
<!--- Then we sleep for an interval - this is to stop us hogging server resources --->
<cfset sleep(2000) />
<cfelse>
<!--- Nothing in the queue, so sleep for a while... --->
<cfset sleep(10000) />
</cfif>
</cfloop>
</cfthread>
谁能告诉我我是否使用了不正确的范围或其他东西?有没有办法强制清理我的临时对象?我认为调用显式垃圾收集是行不通的,因为它无论如何都没有被清理。
只有当我们从 CF8 迁移到 CF9 时,这才似乎成为问题。
感谢所有帮助 - 我真的很想保留这种线程方法,而不是将其作为计划任务或其他东西运行。
谢谢,夏兰。
最佳答案
我可以提供两条建议。一个非常不确定,但另一个非常可靠。
有人告诉我,虽然我没有测试过,但您可以通过将空字符串分配给对象的变量名来提前将对象释放到垃圾回收中。本质上,您正在减少对象上的指针计数。我只在 CF6.1 环境中看到过这样做,所以我真的不确定它是否适用,假设它确实有效。
如果是我,我会研究的是计划任务。它不会经常运行(我认为最短等待时间是 1 分钟,IIRC),但应该在任务终止时释放调用使用的所有内存。基本上,您只需删除外循环和 sleep(),然后像正常调用一样调用该页面。我不完全确定您的队列是如何工作的,但由于它位于应用程序范围内,只要您的任务位于应用程序树中(或包含该应用程序文件),您仍然应该能够访问它。
根据您的其他评论,听起来这不是共享环境。您运行计划任务是否还有其他障碍?
关于java - ColdFusion 9 - 我的线程占用内存 - 我怎样才能停止它?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3573478/