缓存响应时发生资源泄漏
Resource leak when caching a response
我正在使用 netty 4.0.26.Final 开发代理服务器(基于 HexDumpProxy 示例),它可以缓存对某种类型请求的响应。因此,当接收到其响应被缓存的请求时,它将 return 缓存响应而不遵循管道到主机。
现在我有执行它的逻辑 运行 没有错误但是当我将 ResourceLeakDetector 激活到偏执狂级别时,由于存储在缓存。
这是我的处理程序在 HexDumpProxyFrontendHandler 之前的管道中的代码。插入缓存的ResultMessage实现了一个ByteBufHolder。
public class CacheHandler extends ChannelDuplexHandler {
private final ConcurrentHashMap<String, ResultMessage> cache;
private String cacheKey;
public CacheHandler(ConcurrentHashMap<String, ResultMessage> cache) {
this.cache = cache;
}
@Override
public void channelRead(final ChannelHandlerContext ctx, Object msg) throws Exception {
// Check if the message is an instance of Message, if not we can not cache its value
if (msg instanceof Message) {
Message myMsg = (Message) msg;
// Check if there is a cache key for this Message,
// again if no cache key is available the message is not suitable for caching
cacheKey = myMsg.getCacheKey();
if (cacheKey != null) {
ResultMessage resultMsg = cache.get(cacheKey);
if (resultMsg != null) {
// Response is actually cached and returned directly
ctx.writeAndFlush(resultMsg.retain()).addListener(ChannelFutureListener.CLOSE);
} else {
// Response is not cached yet, it is necessary to follow the pipeline to the host
ctx.fireChannelRead(msg);
}
} else {
ctx.fireChannelRead(msg);
}
} else {
ctx.fireChannelRead(msg);
}
}
@Override
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
// Check if it is a response to a request suitable for caching
if (cacheKey != null && msg instanceof ResultMessage) {
ResultMessage resultMsg = (ResultMessage) msg;
cache.put(cacheKey, resultMsg.retain());
}
ctx.write(msg, promise);
}
}
这是记录的错误
[nioEventLoopGroup-3-4] ERROR io.netty.util.ResourceLeakDetector - LEAK: ByteBuf.release() was not called before it's garbage-collected. See http://netty.io/wiki/reference-counted-objects.html for more information.
任何避免此资源泄漏的帮助将不胜感激。
我意识到我在返回缓存消息之前并没有释放频道中收到的原始消息。所以解决方法是在返回前释放它。
...
// Response is actually cached and returned directly
myMsg.release();
ctx.writeAndFlush(resultMsg.retain()).addListener(ChannelFutureListener.CLOSE);
...
我正在使用 netty 4.0.26.Final 开发代理服务器(基于 HexDumpProxy 示例),它可以缓存对某种类型请求的响应。因此,当接收到其响应被缓存的请求时,它将 return 缓存响应而不遵循管道到主机。
现在我有执行它的逻辑 运行 没有错误但是当我将 ResourceLeakDetector 激活到偏执狂级别时,由于存储在缓存。
这是我的处理程序在 HexDumpProxyFrontendHandler 之前的管道中的代码。插入缓存的ResultMessage实现了一个ByteBufHolder。
public class CacheHandler extends ChannelDuplexHandler {
private final ConcurrentHashMap<String, ResultMessage> cache;
private String cacheKey;
public CacheHandler(ConcurrentHashMap<String, ResultMessage> cache) {
this.cache = cache;
}
@Override
public void channelRead(final ChannelHandlerContext ctx, Object msg) throws Exception {
// Check if the message is an instance of Message, if not we can not cache its value
if (msg instanceof Message) {
Message myMsg = (Message) msg;
// Check if there is a cache key for this Message,
// again if no cache key is available the message is not suitable for caching
cacheKey = myMsg.getCacheKey();
if (cacheKey != null) {
ResultMessage resultMsg = cache.get(cacheKey);
if (resultMsg != null) {
// Response is actually cached and returned directly
ctx.writeAndFlush(resultMsg.retain()).addListener(ChannelFutureListener.CLOSE);
} else {
// Response is not cached yet, it is necessary to follow the pipeline to the host
ctx.fireChannelRead(msg);
}
} else {
ctx.fireChannelRead(msg);
}
} else {
ctx.fireChannelRead(msg);
}
}
@Override
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
// Check if it is a response to a request suitable for caching
if (cacheKey != null && msg instanceof ResultMessage) {
ResultMessage resultMsg = (ResultMessage) msg;
cache.put(cacheKey, resultMsg.retain());
}
ctx.write(msg, promise);
}
}
这是记录的错误
[nioEventLoopGroup-3-4] ERROR io.netty.util.ResourceLeakDetector - LEAK: ByteBuf.release() was not called before it's garbage-collected. See http://netty.io/wiki/reference-counted-objects.html for more information.
任何避免此资源泄漏的帮助将不胜感激。
我意识到我在返回缓存消息之前并没有释放频道中收到的原始消息。所以解决方法是在返回前释放它。
...
// Response is actually cached and returned directly
myMsg.release();
ctx.writeAndFlush(resultMsg.retain()).addListener(ChannelFutureListener.CLOSE);
...