编解码技术,其实就是指java的序列化技术。序列化的目的无非是两个:网络传输和对象持久化。
对于netty来说,重点在于网络传输。
java序列化仅仅是java编解码技术的一种,由于它的种种缺陷,衍生出了多种编解码技术和框架。比如
MessagePack
Google Protobuf
JBoss Marshalling
本文都会一一介绍
在远程服务调用(RPC)时,很少直接使用Java序列化进行消息的编解码和传输,这是什么原因呢?
java序列化的缺点:
1,无法跨语言:比如对方是c写的程序,c编码后传过来,将不能解码为java对象。事实上,目前几乎所有流行的Java RPC通信框架,都没有使用Java序列化作为编解码框架,原因就在于它无法跨语言,而这些RPC框架往往需要支持跨语言调用。
2,序列化后的码流太大:
3,序列化性能太低:
========================================
=======JBoss Marshalling是一个Java对象的序列化API包,修正了JDK自带的序列化包的很多问题,但又保持跟Serializable接口的兼容。
这种编码技术更多的是在JBoss内部使用,应用范围有限。
优点是api非常简单,现在用的人越来越多了。
直接上代码
下面是编码器跟解码器的工厂类。不用太纠结具体的实现,直接copy就行。
public final class MarshallingCodeCFactory {
/**
* 创建Jboss Marshalling解码器MarshallingDecoder
* @return MarshallingDecoder
*/
public static MarshallingDecoder buildMarshallingDecoder() {
//首先通过Marshalling工具类的精通方法获取Marshalling实例对象 参数serial标识创建的是java序列化工厂对象。
final MarshallerFactory marshallerFactory = Marshalling.getProvidedMarshallerFactory("serial");
//创建了MarshallingConfiguration对象,配置了版本号为5
final MarshallingConfiguration configuration = new MarshallingConfiguration();
configuration.setVersion(5);
//根据marshallerFactory和configuration创建provider
UnmarshallerProvider provider = new DefaultUnmarshallerProvider(marshallerFactory, configuration);
//构建Netty的MarshallingDecoder对象,俩个参数分别为provider和单个消息序列化后的最大长度
MarshallingDecoder decoder = new MarshallingDecoder(provider, 1024);
return decoder;
}
/**
* 创建Jboss Marshalling编码器MarshallingEncoder
* @return MarshallingEncoder
*/
public static MarshallingEncoder buildMarshallingEncoder() {
final MarshallerFactory marshallerFactory = Marshalling.getProvidedMarshallerFactory("serial");
final MarshallingConfiguration configuration = new MarshallingConfiguration();
configuration.setVersion(5);
MarshallerProvider provider = new DefaultMarshallerProvider(marshallerFactory, configuration);
//构建Netty的MarshallingEncoder对象,MarshallingEncoder用于实现序列化接口的POJO对象序列化为二进制数组
MarshallingEncoder encoder = new MarshallingEncoder(provider);
return encoder;
}
}
在服务端和客户端需要把编解码器加上
ServerBootstrap serverBootstrap = new ServerBootstrap();
serverBootstrap.group(bossGroup,workGroup).//把两个工作线程组加入进来
channel(NioServerSocketChannel.class)//使用哪个channel
.childHandler(new ChannelInitializer<SocketChannel>() {//去绑定具体的事件处理器
@Override
protected void initChannel(SocketChannel socketChannel) throws Exception {
socketChannel.pipeline().addLast(MarshallingCodeCFactory.buildMarshallingDecoder());
socketChannel.pipeline().addLast(MarshallingCodeCFactory.buildMarshallingEncoder());
socketChannel.pipeline().addLast(new ServerHandler());
}
})
值得注意的是,Marshalling编解码技术顺带着解决了tcp拆包粘包的问题,不用我们再去单独解决。