InputStream的read()方法

本文探讨了 Java 中 InputStream 类的使用方法,特别是 read 方法在实际应用中的表现,包括如何确保正确读取指定长度的数据。
InputStream此抽象类是表示字节输入流的所有类的超类。

          我们从输入流中读取数据最常用的方法基本上就是如下3read()方法了:

         1 read()方法,这个方法从输入流中读取数据的下一个字节。返回 0 255 范围内的 int 字节值。如果因为已经到达流末尾而没有可用的字节,则返回值 -1

         2read(byte[] b,int off,int len)方法,将输入流中最多 len 个数据字节读入 byte 数组。尝试读取len 个字节,但读取的字节也可能小于该值。以整数形式返回实际读取的字节数。

         3read(byte[] b)方法,从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b 中。以整数形式返回实际读取的字节数。

         第一个方法典型的确定就是处理效率低,不是某些特殊情况,很少使用它,下面说说第2个方法跟第3个方法,第3个方法的本本质其实就是第2个方法的特殊情况,效果等同于:

         read(b, 0,b.length)

所以这里把他们放着一起讨论。

         从第2个方法的API文档说明来看:“将输入流中最多len 个数据字节读入 byte 数组。尝试读取len 个字节,但读取的字节也可能小于该值。以整数形式返回实际读取的字节数。”,最多读取len个字节,这究竟是何意?API文档并没有详细说明。是不是就意味着有可能(注意这里是有可能而不是一定,)读取不到len个字节呢?答案是“是的”。虽然造成这种情况的原因是什么个人并不知道,但是我们可以通过例子来发现这种情况,下面是源代码(由于只是简单的示例,所以代码也就随便写了):

 

ServerSocket端:

 java代码:

            package myspider;  
      
    import java.io.File;  
    import java.io.FileOutputStream;  
    import java.io.IOException;  
    import java.io.InputStream;  
    import java.net.ServerSocket;  
    import java.net.Socket;  
      
    /** 
     * 
     * @author mark 
     */  
    public class MyServerSocket {  
      
        public static void main(String[] args) throws IOException {  
            ServerSocket ss = new ServerSocket(8888);  
            System.out.println("runing");  
            while (true) {  
                byte[] b = new byte[22480];  
                int readBytes = 0;  
                Socket s = ss.accept();  
                InputStream is = s.getInputStream();  
                while (readBytes < 22480) {  
                    int read = is.read(b, readBytes, 22480 - readBytes);  
                    System.out.println(read);  
                    if (read == -1) {  
                        break;  
                    }  
                    readBytes += read;  
                }  
                File f = new File("F:\\project\\bocln_nacec\\xml\\ey.xml");  
                if (!f.exists()) {  
                    f.createNewFile();  
                    System.out.println("creat " + f.toString());  
                }  
                FileOutputStream fos = new FileOutputStream(f);  
                fos.write(b, 0, readBytes);  
                fos.flush();  
                fos.close();  
                System.out.println("complete");  
                is.close();  
                s.close();  
            }  
        }  
    }  

Socket端:

 

    package myspider;  
      
    import java.io.File;  
    import java.io.FileInputStream;  
    import java.io.IOException;  
    import java.io.InputStream;  
    import java.io.OutputStream;  
    import java.net.Socket;  
    import java.net.UnknownHostException;  
      
    /** 
     * 
     * @author mark 
     */  
    public class MySocket {  
      
        public static void main(String[] args) throws UnknownHostException, IOException {  
            Socket s = new Socket("127.0.0.1", 8888);  
            OutputStream os = s.getOutputStream();  
            File f = new File("F:\\project\\bocln_nacec\\xml\\ye.xml");  
            InputStream is = new FileInputStream(f);  
            byte[] b = new byte[22480];  
            int i = is.read(b);  
            is.close();  
            os.write(b, 0, i);  
            os.flush();  
            os.close();  
            s.close();  
        }  
    }  

 

先运行MyServerSocket,让后多次运行MySocket,这是控制台的输出结果(ye.xml文件长度为20389):

  

java代码:

    runing  
    20389  
    -1  
    creat F:\project\bocln_nacec\xml\ey.xml  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    8760  
    11629  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    3760  
    620  
    16009  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    8760  
    11629  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    8760  
    11629  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    8760  
    11629  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  
    20389  
    -1  
    complete  

         通过观察发现,在大多数情况下,我们能够用is.read(b, readBytes, 22480 - readBytes)一次性就读完整个文件,但是还是有极少数情况,我们需要两次(如36、37两行)甚至两次以上(如58、59、60)调用is.read(b, readBytes, 22480 - readBytes)方法才能把整个文件读取完。这里由于文件最长只有20389,所以我们能读到的最大字节数也就是20389而不会是22480了。

         那么我们怎样写代码才能保证在数据流没有到达末尾的情况下读取到自己想要的长度的字节数据呢?我们可以这样写:  

 

java代码:

    int readBytes=0;  
      
    Byte[] b=new byte[1024]//1024可改成任何需要的值  
      
    int len=b.length;  
      
    while (readBytes < len) {  
      
                    int read = is.read(b, readBytes, len - readBytes);  
      
                   //判断是不是读到了数据流的末尾 ,防止出现死循环。  
      
                    if (read == -1) {  
      
                        break;  
      
                    }  
      
                    readBytes += read;  
      
                }  


评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值