mina自定义编解码器接收处理byte数组(同时解决数据传输中的粘包、缺包问题)

本文介绍了一种自定义传输协议的实现方法,通过使用Apache Mina框架完成字节数组的编码与解码过程。该协议规定字节数组的前四个字节为后续数据的长度,随后跟随实际数据内容。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我们在自定义传输协议时,通常都是采用字节数组的方式进行传送,如何正确接收和解码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()));
### 关于 Java Mina 框架中的 ASCII 编码使用 Java Mina 是一个网络应用程序框架,用于开发高性能的服务器和客户端程序[^1]。它提供了抽象层来处理复杂的 I/O 操作,简化了开发者的工作流程。 #### ASCII 编码概述 ASCII (American Standard Code for Information Interchange) 是一种字符编码标准,广泛应用于计算机通信领域。在 Java 中,可以通过 `String.getBytes("US-ASCII")` 方法将字符串转换为 ASCII 字节流,也可以通过 `new String(byteArray, "US-ASCII")` 将字节数组解码回字符串形式[^1]。 #### 在 Java Mina 中的应用场景 当使用 Java Mina 处理基于文本协议的数据传输时,通常会涉及 ASCII 编码。以下是几个常见的应用场景: 1. **数据编解码器实现** 在 Mina 的架构中,`ProtocolDecoder` 和 `ProtocolEncoder` 负责消息的序列化与反序列化操作。如果应用需要支持 ASCII 文本协议,则可以在这些组件中定义如何将消息对象映射到 ASCII 数据以及反之的操作。 下面是一个简单的 ASCII 解码器示例: ```java public class AsciiTextDecoder extends ProtocolDecoderAdapter { @Override public void decode(IoSession session, IoBuffer in, ProtocolDecoderOutput out) throws Exception { byte[] data = new byte[in.remaining()]; in.get(data); String asciiMessage = new String(data, "US-ASCII"); out.write(asciiMessage); // 输出解析后的 ASCII 字符串 } } ``` 2. **过滤非 ASCII 字符** 如果接收到的数据可能含非法字符(即不在 ASCII 表范围内的字符),可以设计逻辑对其进行过滤或替换。例如,在接收端验证输入是否完全由合法的 ASCII 值组成。 3. **性能优化考虑** 使用固定长度的消息头或者分隔符作为帧边界标记有助于提高效率并减少错误率。对于纯 ASCII 协议而言,“\r\n” 或其他特殊控制字符常被用来分割不同记录项。 4. **异常情况处理** 需要注意的是,实际项目里可能会遇到各种各样的问题比如断现象等都需要妥善解决才能保证整个系统的稳定性运行良好。 ```java public class FixedLengthAsciiEncoder extends ProtocolEncoderAdapter { private final int maxLength; public FixedLengthAsciiEncoder(int maxLength){ this.maxLength = maxLength; } @Override public void encode(IoSession session, Object message, ProtocolEncoderOutput out)throws Exception{ if(message instanceof String && ((String)message).length()<=maxLength){ ByteBuffer buffer=ByteBuffer.allocate(maxLength); buffer.put(((String)message).getBytes("US-ASCII")); while(buffer.position()<maxLength){buffer.put((byte)' ');} buffer.flip(); out.write(new DefaultWriteFuture(session,new BufferByteBufAllocator().allocate(buffer))); }else throw new IllegalArgumentException("Invalid Message Length."); } } ``` 以上代码片段展示了如何创建一个定长的 ASCII 编码器,确保每条发送出去的信息都填充至指定的最大长度。 ### 注意事项 尽管上述方法适用于大多数基本需求,但在更复杂的情况下还需要进一步调整策略以满足特定业务的要求。此外,考虑到跨平台兼容性和国际化因素的影响,在某些场合下单纯依赖 ASCII 可能不够全面,这时就需要引入 UTF-8 等更加灵活丰富的编码方式[^1]。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值