Java网络编程——第二、三章 流

Java的IO建立在流基础上,分为字节流与字符流两大类,注意流是同步的,并且一般IO速度较慢,尽量将IO放在一个单独的线程中

输出流
java.io.OutputStream: public abstract class OutputStream
需要注意的是其中的一个基本方法
public abstract void write(int b) htrows IOException,接受一个0~255之间的无符号整数,将该字节写入输出流,这里使用 int 参数是因为Java不支持无符号数,对于超过255的数字,jdk的做法是仅写入低8位(int -> byte);


package ch2.GernerateCharacters;
import java.io.IOException;
import java.io.OutputStream;
public class GernerateCharacters {

     public static void main(String[] args) throws IOException {
           // TODO Auto-generated method stub
           OutputStream out = System.out;
           generateCharacters2(out);
     }
     //一个字节一个字节的输出
     public static void generateCharacters1(OutputStream out) throws IOException{
           int firstPrintableCharacter = 33; // ASCII = 33 即 ‘!’
           int numberOfPrintableCharacters = 94;
           int numberOfChatactersPerLine = 72;

           int start = firstPrintableCharacter;
           int lines = 15;
           while(true ) {
                for(int i = start; i < start + numberOfChatactersPerLine; i++) {
                     out.write((
                                (i - firstPrintableCharacter) % numberOfPrintableCharacters)
                                + firstPrintableCharacter);
                }
                out.write('\r');
                out.write('\n');
                start = ((start + 1) - firstPrintableCharacter) % numberOfPrintableCharacters
                           + firstPrintableCharacter;
                lines--;
                if(lines == 0) {
                     break;
                }
           }
           //后序处理
           //刷新流
           out.flush();
           //关闭流
           if(out != null) {
                out.close();
           }
           out = null;
     }

     // 单个字节的输出效率不高,对于以太网的每个TCP分片,至少有40个字节的路由和纠错开销,因此,常见的做法是将数据缓存起来
     // 等到一定的数量或者一定时间时,统一输出;需要注意的是,由于采用了缓冲方式,有可能存在缓冲区不满而导致数据不发送,
     // 因此,需要在有必要的时候使用flush方法刷新流,同时在关闭流的时候,也需要进行此操作;
     // 关闭流的方式应该在finally中完成,为了避免空指针异常,在关闭前需要检查流是否为空!

     // note: Java7 引入了带资源的try语句,可以不显式的关闭流,而是自动关闭
     // 这里要求流申明了AutoCloseable接口,而绝大多数流都申明了该接口,除了JavaMail Transport对象
     public static void generateCharacters2(OutputStream out) throws IOException{
           int firstPrintableCharacter = 33; // ASCII = 33 即 ‘!’
           int numberOfPrintableCharacters = 94;
           int numberOfChatactersPerLine = 72;

           int start = firstPrintableCharacter;
           byte[] buffer = new byte[numberOfChatactersPerLine + 2];  //2是给回车和换行符的
           int lines = 15;
           while(true) {
                for(int i = start; i < start + numberOfChatactersPerLine; i++) {
                     buffer[i - start] = (byte)((i - firstPrintableCharacter) % numberOfPrintableCharacters
                                + firstPrintableCharacter);
                }
                buffer[72] = (byte) ('\r');
                buffer[73] = (byte) ('\n');
                out.write(buffer);
                start = ((start + 1) - firstPrintableCharacter) % numberOfPrintableCharacters
                           + firstPrintableCharacter;
                lines--;
                if(lines == 0) {
                     break;
                }
           }
           out.flush();
           if(out != null) {
                out.close();
           }
           out = null;
     }
}
输入流
Java.io.InputStream: public bastract class InputStream
需要注意的是其中的一个基本方法
public abstract int read() htrows IOException
从输入流中读取一个字节的数据,作为0~255之间的int值返回,流结束则返回-1;注意返回的是int型数据,必要时需要进行类型转换;但是int字节在转换是可能出现负数,可以使用如下方式将有符号字节转为无符号字节
int i = b >= 0 ? b : b + 256;

和输出流类似,单个字节的读取效率低下,可以使用缓冲,使用 int read(byte[] input) 读取多个字节,由于传输延时、流关闭的等原因,该方法返回的数实际读取的字节数 。由于CPU速度远快于网络速度,程序很容易在数据全部达到前清空缓冲区,而该方法在试图读取为空但任打开的流时,返回0,而int read()方法则会阻塞运行的线程;同时,在流关闭时,将返回-1,但缓冲区中只会存放实际数据,-1不会进入缓冲区

int bytesToRead = 1024;
int bytesRead = 0;
byte[] input = new byte[bytesToRead];
while(bytesRead < bytesToRead) {
     int result = in.read(input, bytesRead, bytesToRead - bytesRead); 
     // 考虑到剩下的数据因为刘关闭而不可达
     if(result == -1) {
          break;
     }
     bytesRead += result;
}
当然,还可以使用available()方法获取在不阻塞的前提下可以读取的字节数,他返回可以读取的最少字节数,需要注意的是,在流的最后available会返回0,而read(byte[] input, int offset, int length)在length为0的时候,会返回0而不是认为流已经关闭
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值