缓冲区下溢异常

Posted

技术标签:

【中文标题】缓冲区下溢异常【英文标题】:BufferUnderflowException 【发布时间】:2013-02-11 06:43:08 【问题描述】:

在蓝牙应用程序中,我正在从外部设备接收字节 [] 中的数据。 然后将数据放入 ByteBuffer 以供进一步使用

如下图

        ByteBuffer localByteBuffer = null;
        byte[] arrayOfByte = new byte[4096];

          int bytes = 0;
          Log.d("ZeoTest", "++++ Listening...");
          while (true) 

              try 
                    bytes = in.read(arrayOfByte);
                    localByteBuffer = ByteBuffer.wrap(arrayOfByte);

                    localByteBuffer.get(arrayOfByte, 0, arrayOfByte.length);

                    Log.d("Zeo", "input :"+((localByteBuffer))); 
                    String data =bytesToHex(arrayOfByte) ;
                    Log.d("Zeo", "input stream :"+(new String(data))); 
                    Log.d("ZeoTest", "++++ Read "+ bytes +" bytes");

               catch (IOException e) 
                  e.printStackTrace();
                  try  Thread.sleep(0);  catch (InterruptedException ie) 
               Log.d("ZeoTest", "++++ Done: test()");

              onReceive(localByteBuffer);   

在 onReceive 方法中检查标头是否有效...

如下图

  public static boolean isValidHeader(EnhancedByteBuffer paramEnhancedByteBuffer)
  
    boolean bool = false;
    if (paramEnhancedByteBuffer.remaining() < 12);
    while (true)
    

      paramEnhancedByteBuffer.order(ByteOrder.LITTLE_ENDIAN);
      int i = paramEnhancedByteBuffer.position();
      long l = paramEnhancedByteBuffer.getUint32();
      paramEnhancedByteBuffer.getUint16();
      int j = paramEnhancedByteBuffer.getUint8();
      int k = paramEnhancedByteBuffer.getUint8();
      paramEnhancedByteBuffer.getBoolean();
      paramEnhancedByteBuffer.getUint8();
      int m = paramEnhancedByteBuffer.getUint16();
      paramEnhancedByteBuffer.position(i);
      if ((l == 1196641608L) && (j <= 2) && (k < 14) && (m == MessageType.getMessageContentSize(k)))
        bool = true;
           return bool;
    
  

但在 getUint32 方法的流异常下获取缓冲区

可能是什么原因...????

logcat....

   FATAL EXCEPTION: main
java.nio.BufferUnderflowException
at java.nio.HeapByteBuffer.getInt(HeapByteBuffer.java:117)
at com.myzeo.zeo.utility.EnhancedByteBuffer.getUint32(EnhancedByteBuffer.java:87)
at com.myzeo.bluetooth.ZeoTest.isValidHeader(ZeoTest.java:195)
at com.myzeo.bluetooth.ZeoTest.isValidHeader(ZeoTest.java:183)
at com.myzeo.bluetooth.ZeoTest.onReceive(ZeoTest.java:153)
at com.myzeo.bluetooth.ZeoTest.setup(ZeoTest.java:125)
at com.myzeo.bluetooth.ZeoTest.access$0(ZeoTest.java:64)
at com.myzeo.bluetooth.ZeoTest$1.onClick(ZeoTest.java:47)   
at android.view.View.performClick(View.java:2629)
at android.view.View$PerformClick.run(View.java:9374)
at android.os.Handler.handleCallback(Handler.java:587)
at android.os.Handler.dispatchMessage(Handler.java:92)
at android.os.Looper.loop(Looper.java:123)
at android.app.ActivityThread.main(ActivityThread.java:3683)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:507)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
at dalvik.system.NativeStart.main(Native Method)

【问题讨论】:

原因是当我将数据获取到字节缓冲区时(例如:5ytes).. 位置将是 5... 所以在传递 bytebuffer 以逐字节访问数据之前将位置设置为 0.. 即..localByteBuffer.position(0); 【参考方案1】:

可能的原因是您从缓冲区读取的字节多于可用字节。在调试器中每次 read() 后检查缓冲区的位置,以查看哪个调用使缓冲区超出了应有的程度(可能是 readBoolean())

【讨论】:

【参考方案2】:

您没有考虑阅读头的位置

如果你想得到一个整数值,而读头在一个容量为4的ByteBuffer的位置4,那么你会得到一个BufferUnderflowException,因为他不能再读取4个字节来获取整数值被退回。

你必须给他阅读位置(非持久性):getInt( position ); 例如:getInt(0);

【讨论】:

以上是关于缓冲区下溢异常的主要内容,如果未能解决你的问题,请参考以下文章

最常见到的runtime exception 异常

最常见到的runtime exception 异常

underflow overflow 下溢和上溢

为啥缓冲区未满时字节缓冲区会出现缓冲区溢出异常

批评我的非侵入式堆调试器

整个缓冲区初始化导致异常