我有一个基于 Kemal 的 RESTful Web 服务,它返回“非常大”(从 10 到 17M 大小)的 JSON 数据块,这些数据是由“大”哈希结构中的 to_json 方法生成的。
根据 GC 警告消息,我的代码“可能导致内存泄漏”,而我自己的测量结果表明内存在应用程序运行时“泄漏”。
所以,我认为,手动释放为哈希分配的内存和它的 JSON 字符串表示会很好,但我不知道如何做到这一点:我对不良记录的 GC.free 方法的实验没有成功,我没有'不知道往什么方向继续我的调查...
请告诉我我该怎么做才能避免内存泄漏?
你可以在这里https://github.com/DRVTiny/Druid/blob/master查看我非常简单的应用程序(实际上它是在封闭的公司网段内开发的)的不是非常新鲜但通常是实际的版本/src/druid_mp.cr
导致内存泄漏的代码:
get "/service/:serviceid" do |env|
if (svcid = env.params.url["serviceid"]) && svcid.is_a?(String) && svcid =~ /^s?\d+$/
druid.svc_branch_get((svcid[0] == 's' ? svcid[1..-1] : svcid).to_i).to_json
else
halt env, status_code: 404, response: %q({"error": "Wrong service identificator"})
end
rescue ex
halt env, status_code: 503, response: {"error": "Unhandled exception #{ex.message}"}.to_json
end
PS 我在每个用户请求之后插入了 after_all 钩子执行 GC.collect 。不知道,也许这可以解决我的问题(但我认为这根本不是正确的方法)。
UPD:在我将 GC.collect 添加到 after_all Kemal 钩子之后 - 内存泄漏消失了。但是全局 GC.collect 可能太慢了,而且据我所知,它会阻塞所有光纤和 socket.accept()。请让我知道我是否弄错了。