Java网络编程从入门到精通(34):读写缓冲区中的数据---使用get和put方法按顺序读写单个数据

简介:
对于缓冲区来说,最重要的操作就是读写操作。缓冲区提供了两种方法来读写缓冲区中的数据:get put 方法和array 方法。而get put 方法可以有三种读写数据的方式:按顺序读写单个数据、在指定位置读写单个数据和读写数据块。除了上述的几种读写数据的方法外,CharBuffer 类还提供了用于专门写字符串的put append 方法。在本文及后面的文章中将分别介绍这些读写缓冲区的方法。
虽然使用allocate 方法创建的缓冲区并不是一次性地分配内存空间,但我们可以从用户地角度将一个缓冲区想象成一个长度为capacity 的数组。当缓冲区创建后,和数组一样,缓冲区的大小(capacity 值)将无法改变,也无法访问缓冲区外的数据。如下面的代码创建了一个大小为6 的字节缓冲区。
ByteBuffer byteBuffer  =  ByteBuffer.allocate( 6 );
对于byteBuffer 来说,只能访问属于这个缓冲区的六个字节的数据,如果超过了这个范围,将抛出一个BufferOverflowException 异常,这是一个运行时错误,因为这个错误只能在程序运行时被发现。
既然缓冲区和数组类似,那么缓冲区也应该象数组一样可以标识当前的位置。缓冲区的position 方法为我们提供了这个功能。position 方法有两种重载形式,它们的定义如下:
public   final   int  position()
public   final  Buffer position( int  newPosition)
第一个重载形式用来获取缓冲区的当前位置。在创建缓冲区后,position 的初始值是0 ,也就是缓冲区第一个元素的位置。当从缓冲区读取一个元素后,position 的值加1 。我们从这一点可以看出,position 方法返回的位置就是当前可以读取的元素的位置。position 的取值范围从0 capacity – 1 。如果position 的值等于capacity ,说明缓冲区当前已经没有数据可读了。
position 方法的第二个重载形式可以设置缓冲区的当前位置。参数newPosition 的取值范围是0 <= newPosition < capacity 。如果newPosition 的值超出这个范围,position 方法就会抛出一个IllegalArgumentException 异常。
在大多数情况下不需要直接控制缓冲区的位置。缓冲区类提供的用于读写数据的方法可以自动地设置缓冲区的当前位置。在缓冲区类中,get put 方法用于读写缓冲区中的数据。get put 方法的定义如下:
ByteBuffer 类的 get put 方法:
public   abstract   byte  get()            
public   abstract  ByteBuffer put( byte  b) 
IntBuffer 类的 get put 方法:
public   abstract   int  get()            
public   abstract  IntBuffer put( int  i) 
其他五个缓冲区类中的get put 方法定义和上面的定义类似,只是get 方法返回相应的数据类型,而put 方法的参数是相应的数据类型,并且返回值的类型是相应的缓冲区类。
每当put 方法向缓冲区写入一个数据后,缓冲区的当前位置都会加1 。如果缓冲区的当前位置已经等于capacity ,调用put 方法就会抛出一个java.nio.BufferOverflowException 异常。在缓冲区未初赋值的区域将被0 填充。 使用get 方法可以得到缓冲区当前位置的数据,并使缓冲区的当前位置加1 。和put 方法一样,在缓冲区当前位置等于capacity 时使用get 方法也会抛出java.nio.BufferOverflowException 异常。缓冲区的初始状态如图1 所示。

 
 
图1  缓冲区的初始状态
从图1 可以看出,在缓冲区创建之初,当前的位置和缓冲区中的数据都为0 。当使用如下语句向缓冲区中写入数据后,缓冲区当前状态如图2 所示。
 
byteBuffer.put(( byte ) 2 );
byteBuffer.put((
byte ) - 1 );
图2  缓冲区的当前状态
 
当缓冲区的当前位置如图 3 所示时,使用 put get 方法将会抛出上述的 BufferOverflowException 异常。
图3  当前位置处于缓冲区尾
如果要使用get 方法得到缓冲区中的指定数据,必须将缓冲区的当前位置移动到指定的位置,我们可以使用position 方法将当前位置移到缓冲区的任何位置。如下面的代码将图3 所示的缓冲区的当前位置设为2 ,并用get 方法获得位置2 的数据:
byteBuffer.position( 2 );
System.out.println(byteBuffer.get());
上面的代码将输出3 。缓冲区的当前位置为除了使用position 方法,也可以使用rewind 方法将缓冲区的当前位置设为0 rewind 方法的定义如下:
public   final  Buffer rewind()
在图 2 所示的缓冲区状态下调用 rewind 方法,就会得到如图 4 的缓冲区状态。
图4  调用rewind方法后的缓冲区状态
接下来让我们执行如下语句:
System.out.println(byteBuffer.get());
缓冲区的状态将如图 5 所示。
图5  调用get方法后的缓冲区状态
缓冲区除了position capacity 外,还提供了一个标识来限制缓冲区可访问的范围。这个标识就是limit limit position 一样,在缓冲区类中也提供了两个重载方法。用于获得和设置limit 的值。limit 方法的定义如下:

public   final   int  limit()
public   final  Buffer limit( int  newLimit) 
在初始状态下,缓冲区的 limit capacity 值相同。但 limit capacity 的区别是 limit 可以通过 limit 方法进行设置,而 capacity 在创建缓冲区时就已经指定了,并且不能改变。(在上面所讲的 position 方法的 newPosition 参数的取值范围时曾说是 0 <= newPosition < capacity ,其实严格地说,应是 0 <= newPosition < limit limit 的其他性质和 capacity 一样。如在图 5 所示的缓冲区状态中将 limit 的值设为 2 ,就变成了图 6 所示的状态。
图6  将limit设为2的缓冲区状态
在这时position 的值等于limit ,就不能访问缓冲区的当前数据,也就是说不能使用get put 方法。否则将抛出BufferOverflowException 异常。由于使用allocate 创建的缓冲区并不是一次性地分配内存空间,因此,可以将缓冲区的capacity 设为很大的值,如10M 。缓冲区过大可能在某些环境中会使系统性能降低(如在PDA 或智能插秧机中),因此,可以使用limit 方法根据具体的情况来限定缓冲区的大小。当然,limit 还可以表示缓冲区中实际的数据量,这将在后面讲解。下面的代码演示了如何使用limit 方法来枚举缓冲区中的数据:
while (byteBuffer.position()  <  byteBuffer.limit())
    System.out.println(byteBuffer.get());
我们还可以用flip hasRemaining 方法来重写上面的代码。flip 方法将limit 设为缓冲区的当前位置。当limit 等于position 时,hasRemaining 方法返回false ,而则返回true  flip hasRemaining 方法的定义如下:

public   final  Buffer flip()
public   final   boolean  hasRemaining() 
     下面的代码演示了如何使用hasRemaining 方法来枚举缓冲区中的数据:
while (byteBuffer.hasRemaining())
    System.out.println(byteBuffer.get());
如果从缓冲区的第一个位置依次使用put 方法向缓冲区写数据,当写完数据后,再使用flip 方法。这样limit 的值就等于缓冲区中实际的数据量了。在网络中传递数据时,可以使用这种方法来设置数据的结束位置。
为了回顾上面所讲内容,下面的代码 总结了创建缓冲区、读写缓冲区中的数据、设置缓冲区的limit position 的方法。
   package  net;
  
  
import  java.nio. * ;
  
  
public   class  GetPutData
  {
      
public   static   void  main(String[] args)
      {
          
//  创建缓冲区的四种方式
          IntBuffer intBuffer  =  IntBuffer.allocate( 10 );
          ByteBuffer byteBuffer 
=  ByteBuffer.allocateDirect( 10 );
          CharBuffer charBuffer 
=  CharBuffer.wrap( " abcdefg " );
          DoubleBuffer doubleBuffer 
=  DoubleBuffer.wrap( new   double [] {  1.1 2.2  });
          
          
//  向缓冲区中写入数据
          intBuffer.put( 1000 );
          intBuffer.put(
2000 );
          
          System.out.println(
" intBuffer的当前位置: "   +  intBuffer.position());
          
          intBuffer.position(
1 );   //  将缓冲区的当前位置设为1
          System.out.println(intBuffer.get());   //  输出缓冲区的当前数据
          
          intBuffer.rewind();  
//  将缓冲区的当前位置设为0
          System.out.println(intBuffer.get());   //  输出缓冲区的当前数据
          
          byteBuffer.put((
byte ) 20 );
          byteBuffer.put((
byte ) 33 );
          byteBuffer.flip();   
//  将limit设为position,在这里是2
          byteBuffer.rewind(); 
          
while (byteBuffer.hasRemaining())   //  枚举byteBuffer中的数据
              System.out.print(byteBuffer.get()  +   "   " );
          
          
while (charBuffer.hasRemaining())   //  枚举charBuffer中的数据
              System.out.print(charBuffer.get()  +   "   " );
  
          
//  枚举doubleBuffer中的数据
           while (doubleBuffer.position()  <  doubleBuffer.limit())
              System.out.print(doubleBuffer.get() 
+   "   " );
  
      }
  }
运行结果:   
intBuffer的当前位置: 2
2000
1000
20   33  a b c d e f g  1.1   2.2
注意:如果必须使用缓冲区的大小来读取缓冲区的数据,尽量不要使用capacity ,而要使用limit 。如尽量不要写成如下的代码:
while (byteBuffer.position()  <  byteBuffer.capacity())
    System.out.println(byteBuffer.get());
这是因为当limit比capacity小时,上面的代码将会抛出一个BufferUnderflowException异常。








 本文转自 androidguy 51CTO博客,原文链接: http://blog.51cto.com/androidguy/214211 ,如需转载请自行联系原作者
相关文章
|
12天前
|
前端开发 JavaScript Java
java常用数据判空、比较和类型转换
本文介绍了Java开发中常见的数据处理技巧,包括数据判空、数据比较和类型转换。详细讲解了字符串、Integer、对象、List、Map、Set及数组的判空方法,推荐使用工具类如StringUtils、Objects等。同时,讨论了基本数据类型与引用数据类型的比较方法,以及自动类型转换和强制类型转换的规则。最后,提供了数值类型与字符串互相转换的具体示例。
|
22天前
|
消息中间件 Java Kafka
在Java中实现分布式事务的常用框架和方法
总之,选择合适的分布式事务框架和方法需要综合考虑业务需求、性能、复杂度等因素。不同的框架和方法都有其特点和适用场景,需要根据具体情况进行评估和选择。同时,随着技术的不断发展,分布式事务的解决方案也在不断更新和完善,以更好地满足业务的需求。你还可以进一步深入研究和了解这些框架和方法,以便在实际应用中更好地实现分布式事务管理。
|
28天前
|
Java
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
67 9
|
19天前
|
JSON Java 程序员
Java|如何用一个统一结构接收成员名称不固定的数据
本文介绍了一种 Java 中如何用一个统一结构接收成员名称不固定的数据的方法。
23 3
|
21天前
|
安全 Java 开发者
Java中WAIT和NOTIFY方法必须在同步块中调用的原因
在Java多线程编程中,`wait()`和`notify()`方法是实现线程间协作的关键。这两个方法必须在同步块或同步方法中调用,这一要求背后有着深刻的原因。本文将深入探讨为什么`wait()`和`notify()`方法必须在同步块中调用,以及这一机制如何确保线程安全和避免死锁。
36 4
|
21天前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
24 4
|
29天前
|
网络协议 Java 物联网
Java网络编程知识点
Java网络编程知识点
41 13
|
19天前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
24 1
|
1月前
|
Java 程序员 容器
Java中的变量和常量:数据的‘小盒子’和‘铁盒子’有啥不一样?
在Java中,变量是一个可以随时改变的数据容器,类似于一个可以反复打开的小盒子。定义变量时需指定数据类型和名称。例如:`int age = 25;` 表示定义一个整数类型的变量 `age`,初始值为25。 常量则是不可改变的数据容器,类似于一个锁死的铁盒子,定义时使用 `final` 关键字。例如:`final int MAX_SPEED = 120;` 表示定义一个名为 `MAX_SPEED` 的常量,值为120,且不能修改。 变量和常量的主要区别在于变量的数据可以随时修改,而常量的数据一旦确定就不能改变。常量主要用于防止意外修改、提高代码可读性和便于维护。
|
1天前
|
SQL 安全 网络安全
网络安全与信息安全:知识分享####
【10月更文挑战第21天】 随着数字化时代的快速发展,网络安全和信息安全已成为个人和企业不可忽视的关键问题。本文将探讨网络安全漏洞、加密技术以及安全意识的重要性,并提供一些实用的建议,帮助读者提高自身的网络安全防护能力。 ####
34 17