Java ByteBuffer在解码为字符串时缺少数据

Java ByteBuffer在解码为字符串时缺少数据,java,string,byte,bytebuffer,Java,String,Byte,Bytebuffer,我在给一个流浪汉读书写字 import org.assertj.core.api.Assertions; import java.io.IOException; import java.net.InetSocketAddress; import java.nio.ByteBuffer; import java.nio.ByteOrder; import java.nio.charset.Charset; import java.nio.charset.CharsetDecoder; impor

我在给一个流浪汉读书写字

import org.assertj.core.api.Assertions;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.nio.charset.CharsetEncoder;

public class Solution{
public static void main(String[] args) throws Exception{

    final CharsetEncoder messageEncoder = Charset.forName("ISO-8859-1").newEncoder();
    String message = "TRANSACTION IGNORED";
    String carrierName= "CARR00AB";
    int messageLength = message.length()+carrierName.length()+8;


    System.out.println(" --------Fill data---------");
    ByteBuffer messageBuffer = ByteBuffer.allocate(4096);
    messageBuffer.order(ByteOrder.BIG_ENDIAN);
    messageBuffer.putInt(messageLength);
    messageBuffer.put(messageEncoder.encode(CharBuffer.wrap(carrierName)));
    messageBuffer.put(messageEncoder.encode(CharBuffer.wrap(message)));
    messageBuffer.put((byte) 0x2b);
    messageBuffer.flip();

    System.out.println("------------Extract Data Approach 1--------");

    CharsetDecoder messageDecoder = Charset.forName("ISO-8859-1").newDecoder();
    int lengthField = messageBuffer.getInt();
    System.out.println("lengthField="+lengthField);
    int responseLength = lengthField - 12;
    System.out.println("responseLength="+responseLength);
    String messageDecoded= messageDecoder.decode(messageBuffer).toString();
    System.out.println("messageDecoded="+messageDecoded);
    String decodedCarrier = messageDecoded.substring(0, carrierName.length());
    System.out.println("decodedCarrier="+ decodedCarrier);
    String decodedBody = messageDecoded.substring(carrierName.length(), messageDecoded.length() - 1);
    System.out.println("decodedBody="+decodedBody);

    Assertions.assertThat(messageLength).isEqualTo(lengthField);
    Assertions.assertThat(decodedBody).isEqualTo(message);
    Assertions.assertThat(decodedBody).isEqualTo(message);

    ByteBuffer messageBuffer2 = ByteBuffer.allocate(4096);
    messageBuffer2.order(ByteOrder.BIG_ENDIAN);
    messageBuffer2.putInt(messageLength);
    messageBuffer2.put(messageEncoder.encode(CharBuffer.wrap(carrierName)));
    messageBuffer2.put(messageEncoder.encode(CharBuffer.wrap(message)));
    messageBuffer2.put((byte) 0x2b);
    messageBuffer2.flip();

    System.out.println("---------Extract Data Approach 2--------");

    byte [] data = new byte[messageBuffer2.limit()];
    messageBuffer2.get(data);
    String dataString =new String(data, "ISO-8859-1");
    System.out.println(dataString);

}
}
它工作得很好,但后来我想重构它,请参阅上面代码中的方法2

    byte [] data = new byte[messageBuffer.limit()];
    messageBuffer.get(data);
    String dataString =new String(data, "ISO-8859-1");
    System.out.println(dataString);

 Output=     #CARR00ABTRANSACTION IGNORED+
你们能帮我解释一下吗

  • 为什么解码时第二种方法中缺少整数

  • 有没有办法在第二种方法中提取整数


  • 好的,那么您正试图从缓冲区读取一个
    int
    ,它占用
    4位
    ,然后在读取
    4位

    我所做的是调用
    messageBuffer2.clear()读取int后解决此问题。这是完整的代码

    System.out.println(messageBuffer2.getInt());
    byte[] data = new byte[messageBuffer2.limit()];
    messageBuffer2.clear();
    messageBuffer2.get(data);
    String dataString = new String(data, StandardCharsets.ISO_8859_1);
    System.out.println(dataString);
    
    输出为:

    35
       #CARR0033TRANSACTION IGNORED+
    

    编辑:所以基本上,当你调用
    clear
    时,它会重置各种变量,也会重置它所处的位置,这就是它的修复方式。

    没有阅读问题,你是否提供了编码?是的,请查看问题中的我的代码好的,我会在我这边编译代码,如果其他人还没有在我之前验证这个问题。你上一个代码放在哪里了?您能提供工作示例吗?请参阅更新,请在执行codeCan之前删除断言。请解释(背后的逻辑)调用“clear”如何解决此问题??