`

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

    博客分类:
  • Java
阅读更多

在家上网赚钱更容易

 对于缓冲区来说,最重要的操作就是读写操作。缓冲区提供了两种方法来读写缓冲区中的数据:getput方法和array方法。而getput方法可以有三种读写数据的方式:按顺序读写单个数据、在指定位置读写单个数据和读写数据块。除了上述的几种读写数据的方法外,CharBuffer类还提供了用于专门写字符串的putappend方法。在本文及后面的文章中将分别介绍这些读写缓冲区的方法。

虽然使用allocate方法创建的缓冲区并不是一次性地分配内存空间,但我们可以从用户地角度将一个缓冲区想象成一个长度为capacity的数组。当缓冲区创建后,和数组一样,缓冲区的大小(capacity值)将无法改变,也无法访问缓冲区外的数据。如下面的代码创建了一个大小为6的字节缓冲区。

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->ByteBuffer byteBuffer = ByteBuffer.allocate(6);

对于byteBuffer来说,只能访问属于这个缓冲区的六个字节的数据,如果超过了这个范围,将抛出一个BufferOverflowException异常,这是一个运行时错误,因为这个错误只能在程序运行时被发现。

既然缓冲区和数组类似,那么缓冲区也应该象数组一样可以标识当前的位置。缓冲区的position方法为我们提供了这个功能。position方法有两种重载形式,它们的定义如下:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->public final int position()
public final Buffer position(int newPosition)

第一个重载形式用来获取缓冲区的当前位置。在创建缓冲区后,position的初始值是0,也就是缓冲区第一个元素的位置。当从缓冲区读取一个元素后,position的值加1。我们从这一点可以看出,position方法返回的位置就是当前可以读取的元素的位置。position的取值范围从0capacity – 1。如果position的值等于capacity,说明缓冲区当前已经没有数据可读了。

position方法的第二个重载形式可以设置缓冲区的当前位置。参数newPosition的取值范围是0 <= newPosition < capacity。如果newPosition的值超出这个范围,position方法就会抛出一个IllegalArgumentException异常。

在大多数情况下不需要直接控制缓冲区的位置。缓冲区类提供的用于读写数据的方法可以自动地设置缓冲区的当前位置。在缓冲区类中,getput方法用于读写缓冲区中的数据。getput方法的定义如下:

ByteBuffer类的getput方法:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->public abstract byte get()            
public abstract ByteBuffer put(byte b)

IntBuffer类的getput方法:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->public abstract int get()            
public abstract IntBuffer put(int i)

其他五个缓冲区类中的getput方法定义和上面的定义类似,只是get方法返回相应的数据类型,而put方法的参数是相应的数据类型,并且返回值的类型是相应的缓冲区类。

每当put方法向缓冲区写入一个数据后,缓冲区的当前位置都会加1。如果缓冲区的当前位置已经等于capacity,调用put方法就会抛出一个java.nio.BufferOverflowException异常。在缓冲区未初赋值的区域将被0填充。使用get方法可以得到缓冲区当前位置的数据,并使缓冲区的当前位置加1。和put方法一样,在缓冲区当前位置等于capacity时使用get方法也会抛出java.nio.BufferOverflowException异常。缓冲区的初始状态如图1所示。

 

图1 缓冲区的初始状态

 

从图1可以看出,在缓冲区创建之初,当前的位置和缓冲区中的数据都为0。当使用如下语句向缓冲区中写入数据后,缓冲区当前状态如图2所示。

 

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->byteBuffer.put((byte)2);
byteBuffer.put((
byte)-1);

 

 

图2  缓冲区的当前状态

 

 

当缓冲区的当前位置如图3所示时,使用putget方法将会抛出上述的BufferOverflowException异常。

 

图3  当前位置处于缓冲区尾

 

如果要使用get方法得到缓冲区中的指定数据,必须将缓冲区的当前位置移动到指定的位置,我们可以使用position方法将当前位置移到缓冲区的任何位置。如下面的代码将图3所示的缓冲区的当前位置设为2,并用get方法获得位置2的数据:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->byteBuffer.position(2);
System.out.println(byteBuffer.get());

上面的代码将输出3。缓冲区的当前位置为除了使用position方法,也可以使用rewind方法将缓冲区的当前位置设为0rewind方法的定义如下:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->public final Buffer rewind()

在图2所示的缓冲区状态下调用rewind方法,就会得到如图4的缓冲区状态。

 

图4  调用rewind方法后的缓冲区状态

接下来让我们执行如下语句:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->System.out.println(byteBuffer.get());

缓冲区的状态将如图5所示。

 

图5  调用get方法后的缓冲区状态

 

缓冲区除了positioncapacity外,还提供了一个标识来限制缓冲区可访问的范围。这个标识就是limitlimitposition一样,在缓冲区类中也提供了两个重载方法。用于获得和设置limit的值。limit方法的定义如下:

 

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->public final int limit()
public final Buffer limit(int newLimit)

在初始状态下,缓冲区的limitcapacity值相同。但limitcapacity的区别是limit可以通过limit方法进行设置,而capacity在创建缓冲区时就已经指定了,并且不能改变。(在上面所讲的position方法的newPosition参数的取值范围时曾说是0 <= newPosition < capacity,其实严格地说,应是0 <= newPosition < limitlimit的其他性质和capacity一样。如在图5所示的缓冲区状态中将limit的值设为2,就变成了图6所示的状态。

 

图6  将limit设为2的缓冲区状态

 

在这时position的值等于limit,就不能访问缓冲区的当前数据,也就是说不能使用getput方法。否则将抛出BufferOverflowException异常。由于使用allocate创建的缓冲区并不是一次性地分配内存空间,因此,可以将缓冲区的capacity设为很大的值,如10M。缓冲区过大可能在某些环境中会使系统性能降低(如在PDA或智能插秧机中),因此,可以使用limit方法根据具体的情况来限定缓冲区的大小。当然,limit还可以表示缓冲区中实际的数据量,这将在后面讲解。下面的代码演示了如何使用limit方法来枚举缓冲区中的数据:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->while(byteBuffer.position() < byteBuffer.limit())
    System.out.println(byteBuffer.get());

我们还可以用fliphasRemaining方法来重写上面的代码。flip方法将limit设为缓冲区的当前位置。当limit等于position时,hasRemaining方法返回false,而则返回true fliphasRemaining方法的定义如下:

 

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->public final Buffer flip()
public final boolean hasRemaining()

    下面的代码演示了如何使用hasRemaining方法来枚举缓冲区中的数据:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->while(byteBuffer.hasRemaining())
    System.out.println(byteBuffer.get());

如果从缓冲区的第一个位置依次使用put方法向缓冲区写数据,当写完数据后,再使用flip方法。这样limit的值就等于缓冲区中实际的数据量了。在网络中传递数据时,可以使用这种方法来设置数据的结束位置。

为了回顾上面所讲内容,下面的代码总结了创建缓冲区、读写缓冲区中的数据、设置缓冲区的limitposition的方法。

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->  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.12.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() 
+ " ");
  
      }
  }

运行结果:  

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->intBuffer的当前位置:2
2000
1000
20 33 a b c d e f g 1.1 2.2

 

注意:如果必须使用缓冲区的大小来读取缓冲区的数据,尽量不要使用capacity,而要使用limit。如尽量不要写成如下的代码:

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->while(byteBuffer.position() < byteBuffer.capacity())
    System.out.println(byteBuffer.get());

这是因为当limit比capacity小时,上面的代码将会抛出一个BufferUnderflowException异常。

在家上网赚钱更容易

分享到:
评论

相关推荐

    JAVA网络编程从入门到精通

    ### JAVA网络编程从入门到精通知识点详解 #### 一、Internet地址概述 互联网中的每一台设备都需要有一个唯一的标识符——IP地址。当前广泛使用的IPv4地址由四个字节组成,而未来的趋势是采用16个字节的IPv6地址。 ...

    jdk6.0从入门到精通-----chapter5网络编程 新I/O(含源码下载)

    在Java编程领域,JDK(Java Development Kit)是开发和运行Java应用程序的...总的来说,"JDK6.0从入门到精通-----chapter5网络编程 新I/O"是一个极好的学习资源,无论你是初学者还是有经验的开发者,都能从中获益良多。

    Java网络编程

    #### 三十三、读写缓冲区中的数据---使用`get`和`put`方法按顺序读写单个数据 `Buffer`类提供了`get`和`put`方法,用于从缓冲区中读取数据或将数据写入缓冲区,支持按顺序读写单个数据元素。 综上所述,Java网络...

    网络编程教程,绝对经典

    #### 三十三、读写缓冲区中的数据---使用get和put方法按顺序读写单个数据 - **数据操作**:使用get()方法读取数据,put()方法写入数据。 - **流程**:确保正确配置缓冲区的模式(read/write)。 通过以上内容,...

    Java NIO学习笔记——ByteBuffer用法

    本文主要关注的是Java NIO中的ByteBuffer,一个关键的数据容器,用于在通道(Channel)和缓冲区(Buffer)之间传输数据。ByteBuffer的用法是Java NIO学习中的核心内容。 首先,我们了解下ByteBuffer的基本概念。...

    java_NIO_入门(良好排版格式).pdf

    缓冲区的关键方法包括 `put()` 和 `get()`,分别用于写入和读取数据。 ##### 2. 通道(Channel) 通道是一种用于连接数据源和目的地的一种方式。通道可以理解为数据流的通道,它可以读取数据也可以写入数据,具有...

    java NIO原理和使用

    3. **散列和聚合**:NIO 提供了对多缓冲区读写的优化,比如可以通过单个操作将多个缓冲区的数据写入通道或从通道读取到多个缓冲区。 4. **内存映射文件**:NIO 支持将文件直接映射到内存中,这在处理大数据量时特别...

    java nio 入门

    2. **分配缓冲区**:通过Buffer子类的allocate()方法创建缓冲区,然后通过put()和get()方法存取数据。 3. **通道与缓冲区交互**:使用通道的read()和write()方法,将数据从通道读入缓冲区或从缓冲区写入通道。 4. ...

    java输入输出整理1

    - `put()`也有五种形式,用于写入单个字节、写入数组、从源缓冲区复制数据、在特定位置写入字节以及相对和绝对写入。 - **类型化的get()和put()方法**: 除了基本的字节操作,ByteBuffer还提供了如getChar()、...

    【IT十八掌徐培成】Java基础第26天-05.ByteBuffer-mark-pos-limit-cap-flip.zip

    比如,从网络接收数据时,我们可能先分配一个足够大的`ByteBuffer`,然后通过网络通道读取数据到缓冲区,使用`flip()`切换到读模式,再从缓冲区读取数据进行处理。 此外,还有其他一些重要的方法,如: - **clear()...

    nio.rar_Java socketA_java nio_java socket a

    - 方法:`put()`(写入)、`get()`(读取)、`flip()`(切换读写模式)、`clear()`(清空缓冲区)、`compact()`(压缩缓冲区,清除已读部分)等。 3. **选择器(Selector)**:选择器允许单个线程管理多个通道,...

    NIO编程实现实例

    缓冲区具有容量、位置和限制等概念,可以通过get和put方法读写数据。 3. **选择器(Selector)**:选择器是NIO中用于多路复用的关键组件。通过向选择器注册通道,可以监控多个通道的事件,如读、写、连接和接受事件...

    Java_TCPIP_Socket编程(中文版)

    - **存储和接收数据**:使用`put()`方法向`Buffer`中写入数据,使用`get()`方法从中读取数据。 - **准备Buffer:clear(),flip(),和rewind()**:这些方法用于准备`Buffer`以便进行读写操作。 - **压缩Buffer中的...

    非阻塞异步传输java网络编程(含源代码)

    - **decode()和encode()方法**:实现对字节缓冲区的编码和解码操作,方便读写。 ##### 3. **实验流程** 1. **客户端**从控制台输入命令(PUTXXX.XXX或GETXXX.XXX),根据命令决定是上传还是下载文件。 2. **服务器**...

    Java IO_NIO

    缓冲区是NIO的核心,它是一种特定类型的数组,如ByteBuffer、CharBuffer等,提供了put和get方法来读写数据。缓冲区有以下特点: 1. 容量:缓冲区有一个固定大小的容量,写入数据不能超过这个容量。 2. 位置和限制:...

    Java语言基础教程-Java NIO流篇1

    另外,`get()`和`put()`方法分别用于读取和写入数据,`read()`和`write()`方法则用于从通道读取到缓冲区,或者将缓冲区的数据写入通道。同时,缓冲区的容量、位置和限制的管理也是这一节的重点内容。 此外,NIO还...

Global site tag (gtag.js) - Google Analytics