我们在Web可访问/控制器目录中有许多cfc,用于处理提交和处理.当没有方法参数直接调用cfc时,服务器开始咀嚼内存.
例如,像http://www.domain.com/controller/LoginController.cfc这样的URL将在浏览器中运行,直到它超时. / CFIDE已被锁定,无法公开访问
所以cfexplorer没有(或不应该)可用.
我们使用FusionReactor来监控我们的实例.我们的服务器设置为20GB的堆空间.在加载应用程序后重新启动时,内存将巡航大约800MB.
在正常流量的情况下,内存将在5GB到10GB之间波动,并定期进行垃圾回收.一段时间后,服务器最终达到98%的容量.它往往会在那里运行
有时可能会持续几个小时甚至几天,直到一些交通高峰推动它并发生一个外部错误.垃圾收集没有恢复内存,也没有活动
FusionReactor报告的长时间运行的线程.只有服务器重启才能恢复它.
使用FusionReactor(我们刚刚安装了这是我最终对此问题有所了解)我正在检查PermGen内存空间并发现它占了
堆的85%.这似乎不对.我执行了内存转储并通过Eclipse将其加载到MAP中进行分析.我发现内存中有10个对象
测量1.7GB(1.7×10约占总堆的85%).这些对象如下所示:
Class Name | Shallow Heap | Retained Heap | Percentage byte[1769628928] @ 0x4d963b198 ...128.................POST......../controller/LoginController.cfc......../controller/LoginController.cfc........173.14.93.66........173.14.93.66........www.domain.com........443........HTTP/1.1.......;D:\websites\domain\system\controller\Lo...| 1,769,628,944 | 1,944 | 8.60%
所以我在我们的一台服务器上重启了CF.检查了FusionReactor并且没有看到内存使用情况.然后去浏览器并首先调用cfc:
http://www.domain.com/controller/LoginController.cfc?method=foo
这导致onMissingMethod处理程序正确地踢和重定向到适当的错误页面而没有服务器效果.
但后来称之为:
http://www.domain.com/controller/LoginController.cfc
导致页面挂起. FusionReactor报告没有活动请求,即使其中一个正在运行,这就是我们无法在问题发生时识别问题的原因.更糟糕的是,刷新内存会使其慢慢增加十分之一个百分点而没有报告活动.服务器上的超时设置为5分钟.我假设它最终会被杀死,然后以1.7GB的身份成为孤儿.这并没有降低服务器的速度,只是将内存以3GB的平均值运行,而垃圾收集没有任何恢复.这似乎解释了为什么随着时间的推移,随机调用这些URL会慢慢咀嚼并保留在内存中.
接下来,我从多个浏览器选项卡调用了URL.这几乎瞬间将记忆飙升至98%.尽管有超过15个浏览器标签在运行,但FusionReactor现在显示了两个长时间运行的请求10秒并攀升.强行杀死线程似乎无能为力.只有服务器重启才能解决问题.
所以现在我已经明确地确定了这个问题(幻像线程在PermGen堆中创建巨大的孤立对象)以及如何复制问题.
如何或为何直接向cfc提出请求我不知道.可能是机器人或偶尔奇怪的浏览器行为.
所有巨大的对象都是jrun.servlet.jrpp.ProxyEndpoint的实例.
具体是什么导致了这个问题,我该如何解决它.
这是运行Java 1.7.0_25的Win2003服务器上的CF9.01 Standard.
谢谢!
解决方法
由于@iKnowKungFoo和大量的实验,我找到了一个解决方法.
将“方法”键/值插入URL范围似乎可以解决问题.需要注意的是,它必须在onRequestStart方法中完成,而不是在onCFCRequest方法中完成.从文档中可以看出,对CFC的调用将直接转到onCFCRequest,但事实并非如此.所有请求首先通过onRequestStart方法.当onRequestStart仅返回时,只有在所需的’method’参数存在时,onCFCRequest才会调用AND.
所以在这种情况下,onCFCRequest从未被调用过,因为’method’参数从未存在过.所以这里是立即在onRequestStart中运行的代码:
<cfif Right(arguments.targetPage,4) IS ".cfc" AND NOT StructKeyExists(URL,"WSDL") AND NOT StructKeyExists(URL,"method") AND NOT StructKeyExists(FORM,"method")> <cfset StructInsert(FORM,"method","")> <cfset StructInsert(URL,"")> </cfif>
这段代码检查所请求页面上的扩展名,如果URL和FORM范围中都不存在方法参数,则会在两者中插入一个空白键/值对,以便进行测量.检查’WSDL’参数就在那里,因为我发现虽然这段代码工作得很好,但突然之间我们已经破解了几个web服务cfc调用.如果对cfc的调用是WebService.cfc?WSDL,则不需要method参数,CF以不同的方式处理整个事件.
因此,插入空的’method’值会导致在onRequsetStart完成时正确调用onCFCRequest.当使用无效的空方法名称调用cfc时,onMissingMethod现在已正确启动.该方法会立即处理错误的页面请求并重定向到自定义错误页面.
自从实施此修复程序以来,我们已经看到所有服务器上的内存使用率从98%一致下降到15%.内存图表显示正在使用和收集的内存的预期锯齿.整体性能已经从平均页面请求时间1200毫秒变为54毫秒,而没有所有这些请求在幕后猖獗.
我仍然希望Adobe能够识别并解决问题.