mina自定义编解码器接收处理byte数组(同时解决数据传输中的粘包、缺包问题)
来源:互联网 发布:教孩子学编程python 编辑:程序博客网 时间:2024/05/19 02:44
我们在自定义传输协议时,通常都是采用字节数组的方式进行传送,如何正确接收和解码byte数组?
假设我们自定义了传输协议: 字节数组的前4个字节是要传输的数据长度,后面跟数据。我们用mina可以这样处理:
1.自定义编码器ByteArrayEncoder.java
import org.apache.mina.core.session.IoSession;
import org.apache.mina.filter.codec.ProtocolEncoderAdapter;
import org.apache.mina.filter.codec.ProtocolEncoderOutput;
/**
* 编码器将数据直接发出去(不做处理)
*/
public class ByteArrayEncoder extends ProtocolEncoderAdapter {
@Override
public void encode(IoSession session, Object message,
ProtocolEncoderOutput out) throws Exception {
out.write(message);
out.flush();
}
}
2.自定义解码器(确保能读取到完整的包)
import org.apache.mina.core.buffer.IoBuffer;
import org.apache.mina.core.session.IoSession;
import org.apache.mina.filter.codec.CumulativeProtocolDecoder;
import org.apache.mina.filter.codec.ProtocolDecoderOutput;
import com.talkweb.meeting.tools.IntByteConvert;
public class ByteArrayDecoder extends CumulativeProtocolDecoder {
@Override
public boolean doDecode(IoSession session, IoBuffer in, ProtocolDecoderOutput out)
throws Exception {
if(in.remaining() > 4){//前4字节是包头
//标记当前position的快照标记mark,以便后继的reset操作能恢复position位置
in.mark();
byte[] l = new byte[4];
in.get(l);
//包体数据长度
int len = MyTools.bytes2int(l);//将byte转成int
//注意上面的get操作会导致下面的remaining()值发生变化
if(in.remaining() < len){
//如果消息内容不够,则重置恢复position位置到操作前,进入下一轮, 接收新数据,以拼凑成完整数据
in.reset();
return false;
}else{
//消息内容足够
in.reset();//重置恢复position位置到操作前
int sumlen = 4+len;//总长 = 包头+包体
byte[] packArr = new byte[sumlen];
in.get(packArr, 0 , sumlen);
IoBuffer buffer = IoBuffer.allocate(sumlen);
buffer.put(packArr);
buffer.flip();
out.write(buffer);
buffer.free();
if(in.remaining() > 0){//如果读取一个完整包内容后还粘了包,就让父类再调用一次,进行下一次解析
return true;
}
}
}
return false;//处理成功,让父类进行接收下个包
}
}
3.编解码工厂类
import org.apache.mina.core.session.IoSession;
import org.apache.mina.filter.codec.ProtocolCodecFactory;
import org.apache.mina.filter.codec.ProtocolDecoder;
import org.apache.mina.filter.codec.ProtocolEncoder;
/**
* @author BruceYang
*
*/
public class ByteArrayCodecFactory implements ProtocolCodecFactory {
private ByteArrayDecoder decoder;
private ByteArrayEncoder encoder;
public ByteArrayCodecFactory() {
encoder = new ByteArrayEncoder();
decoder = new ByteArrayDecoder();
}
@Override
public ProtocolDecoder getDecoder(IoSession session) throws Exception {
return decoder;
}
@Override
public ProtocolEncoder getEncoder(IoSession session) throws Exception {
return encoder;
}
}
4.调用编解码工厂进行编解码
NioSocketAcceptor acceptor = new NioSocketAcceptor();
acceptor.getFilterChain().addLast("mycoder", new ProtocolCodecFilter(new ByteArrayCodecFactory())); 版权声明
假设我们自定义了传输协议: 字节数组的前4个字节是要传输的数据长度,后面跟数据。我们用mina可以这样处理:
1.自定义编码器ByteArrayEncoder.java
import org.apache.mina.core.session.IoSession;
import org.apache.mina.filter.codec.ProtocolEncoderAdapter;
import org.apache.mina.filter.codec.ProtocolEncoderOutput;
/**
* 编码器将数据直接发出去(不做处理)
*/
public class ByteArrayEncoder extends ProtocolEncoderAdapter {
@Override
public void encode(IoSession session, Object message,
ProtocolEncoderOutput out) throws Exception {
out.write(message);
out.flush();
}
}
2.自定义解码器(确保能读取到完整的包)
import org.apache.mina.core.buffer.IoBuffer;
import org.apache.mina.core.session.IoSession;
import org.apache.mina.filter.codec.CumulativeProtocolDecoder;
import org.apache.mina.filter.codec.ProtocolDecoderOutput;
import com.talkweb.meeting.tools.IntByteConvert;
public class ByteArrayDecoder extends CumulativeProtocolDecoder {
@Override
public boolean doDecode(IoSession session, IoBuffer in, ProtocolDecoderOutput out)
throws Exception {
if(in.remaining() > 4){//前4字节是包头
//标记当前position的快照标记mark,以便后继的reset操作能恢复position位置
in.mark();
byte[] l = new byte[4];
in.get(l);
//包体数据长度
int len = MyTools.bytes2int(l);//将byte转成int
//注意上面的get操作会导致下面的remaining()值发生变化
if(in.remaining() < len){
//如果消息内容不够,则重置恢复position位置到操作前,进入下一轮, 接收新数据,以拼凑成完整数据
in.reset();
return false;
}else{
//消息内容足够
in.reset();//重置恢复position位置到操作前
int sumlen = 4+len;//总长 = 包头+包体
byte[] packArr = new byte[sumlen];
in.get(packArr, 0 , sumlen);
IoBuffer buffer = IoBuffer.allocate(sumlen);
buffer.put(packArr);
buffer.flip();
out.write(buffer);
buffer.free();
if(in.remaining() > 0){//如果读取一个完整包内容后还粘了包,就让父类再调用一次,进行下一次解析
return true;
}
}
}
return false;//处理成功,让父类进行接收下个包
}
}
3.编解码工厂类
import org.apache.mina.core.session.IoSession;
import org.apache.mina.filter.codec.ProtocolCodecFactory;
import org.apache.mina.filter.codec.ProtocolDecoder;
import org.apache.mina.filter.codec.ProtocolEncoder;
/**
* @author BruceYang
*
*/
public class ByteArrayCodecFactory implements ProtocolCodecFactory {
private ByteArrayDecoder decoder;
private ByteArrayEncoder encoder;
public ByteArrayCodecFactory() {
encoder = new ByteArrayEncoder();
decoder = new ByteArrayDecoder();
}
@Override
public ProtocolDecoder getDecoder(IoSession session) throws Exception {
return decoder;
}
@Override
public ProtocolEncoder getEncoder(IoSession session) throws Exception {
return encoder;
}
}
4.调用编解码工厂进行编解码
NioSocketAcceptor acceptor = new NioSocketAcceptor();
acceptor.getFilterChain().addLast("mycoder", new ProtocolCodecFilter(new ByteArrayCodecFactory())); 版权声明
0 0
- mina自定义编解码器接收处理byte数组(同时解决数据传输中的粘包、缺包问题)
- mina自定义编解码器接收处理byte数组(同时解决数据传输中的粘包、缺包问题)
- mina自定义编解码器接收处理byte数组(同时解决数据传输中的粘包、缺包问题)
- Mina学习(四):实现自定义编解码器并解决半包,丢包,粘包问题
- Mina中的粘包处理
- Socket通信自定义mina 框架过滤器解析(处理粘包、断包问题)
- Mina传输大数组,多路解码,粘包问题的处理
- Mina通信粘包处理
- mina处理断包和粘包
- mina处理断包和粘包
- mina学习 粘包,断包处理
- mina处理断包和粘包
- mina IoBuffer 处理断包、粘包
- mina解决粘包,找不到解码器,数据帧重传的问题
- Mina 粘包、断包、半包解决
- 自定义Mina编解码器
- 网络游戏数据传输:粘包的处理
- MINA 中的 协议编解码器
- sqlldr导入 sqluldr2导出:
- 安卓v7包去除
- CentOS6.5下安装xampp
- 转,StoryBoard不使用AutoLayout情况下 按比例快速兼容适配iPhone6/6 Plus教程
- Android社交类APP常用的动态消息发布通用模板
- mina自定义编解码器接收处理byte数组(同时解决数据传输中的粘包、缺包问题)
- Centos自动调整错误路径
- (1)fcntl函数说明 F_SETLK/F_SETLKW例子 http://www.jb51.net/article/37671.htm
- hdu 1133 Buy the Ticket
- ---------- 常用链接 ----------
- Struts 2 @ResultPath annotation example
- C Macro UG
- mysql 更改引擎
- 使用Wireshark实时抓包Android