在Netty4中设置ByteBuf ENDIANNESS

时间:2013-11-08 01:57:59

标签: netty

在Netty 3中,我们使用

在每一端强制执行LITTLE_ENDIAN ChannelBuffers
bootstrap.setOption("child.bufferFactory", new HeapChannelBufferFactory(ByteOrder.LITTLE_ENDIAN));

但在Netty 4中,ByteBuf的配置现在似乎是通过ChannelOption.ALLOCATOR:

    bootstrap.option(ChannelOption.ALLOCATOR, someAllocator);

我们真正想做的就是装饰UnpooledByteBufAllocator,但它是最终的和方法 我们需要装饰是受保护的,所以我们不能扩展类或委托给它。我们有 不得不诉诸代理方法:

private static class AllocatorProxyHandler implements InvocationHandler {
    private final ByteBufAllocator allocator;

    public AllocatorProxyHandler(ByteBufAllocator allocator) {
        this.allocator = allocator;
    }

    public static ByteBufAllocator proxy(ByteBufAllocator allocator) {
        return (ByteBufAllocator) Proxy.newProxyInstance(AllocatorProxyHandler.class.getClassLoader(), new Class[]{ByteBufAllocator.class}, new AllocatorProxyHandler(allocator));
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object result = method.invoke(allocator, args);
        if (result instanceof ByteBuf) {
            return ((ByteBuf) result).order(ByteOrder.LITTLE_ENDIAN);
        } else {
            return result;
        }
    }
}

设置Bootstrap选项,如下所示:

    bootstrap.option(ChannelOption.ALLOCATOR, AllocatorProxyHandler.proxy(UnpooledByteBufAllocator.DEFAULT));

还有其他(更好)的方法可以解决这个问题吗?

1 个答案:

答案 0 :(得分:1)

默认情况下,Netty 4.0的ByteBuf是big-endian。您可以使用ByteBuf方法获取order(ByteOrder)的小端视图:

ByteBuf buf = ctx.alloc().buffer();
ByteBuf leBuf = buf.order(ByteOrder.LITTLE_ENDIAN);
leBuf.getByte(...);
...

这是为了避免因将字节顺序保持为状态变量而导致的任何混淆。如果有充分理由说明它必须提供更改默认字节顺序的方法,请告知我们,以便我们重新考虑此决定。