流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象。
根据流向,我们分为输入流和输出流
根据流里数据类型组织结构,我们分为字符流和字节流
字节流 | 字符流 |
一次读入或读出是8位二进制。 | 一次读入或读出是16位二进制。 |
直接对文件操作。 | 流和文件之间有个缓存区。字符流流独立于文件 |
下面程序验证缓冲区的存在。
字节ByteDemo
File file = new File("ByteDemo_01.txt"); OutputStream out = null; String str = "helloworld!!"; try { out = new FileOutputStream(file);//获取输出字节流 out.write(str.getBytes()); } catch (IOException e) { e.printStackTrace(); }
ByteDemo_01.txt内容:
字符流:CharDemo_01
Writer out = null; String str = "helloworld!!"; try { out = new FileWriter(file); out.write(str); }catch(IOException e){ e.printStackTrace(); }
这时CharDemo_01.txt内容为空,对缓冲区没有刷新,字符并没有写入到文件中,在out.write(str);语句后加入out.flush();之后才把字符写入文件。
java处理字节流的超类是InputStream,OutputStream.常用方法见API,
为了弥补字节流没有缓冲区的一些应用场景下的不足,Java给InputStream提供了两种改进方法
-
-
-
- 直接在InputStream上作出程序改进,我们可以自定义缓冲区数组,将读取的内容首先读进自己的缓冲区数组。
-
-
int read(byte[] b, int off, int len)//将字节流读入b数组,off是b数组的起始位置,len是读入的长度,off+len<=b数组的长度
byte[] b = new byte[1024];
in = new FileInputStream(file);
in.read(b,0,1024);//从b[0]对b写入1024字节的数据
-
-
-
- 使用了装饰者类FilterInputStream,其子类BufferedInputStream在程序实现上封装了缓冲区数组,默认大小是8192
-
-
BufferedInputStream bis = new BufferedInputStream(in)
形式上和上面相同。
在访问效率上当第一种设计的方法缓冲区数组长度大于等于8192时和第二种访问效率类似。只是第二种方法提供了mar()和reset()方法,丰富了我们访问数据时所能使用的方法
java处理字符流的超类的Reader,Writer
java处理字符流时默认实现了缓冲区,方法和BufferedInputStream类似,不同的是字符流在读取写入时记得及时刷新缓冲区,防止缓冲区覆盖。
在介绍转换之前首先需要了解解码和编码
解码:String-->byte[] str.getbBytes("GBK")//以GBK形式对原字符串解码
编码:byte[]-->String String str = new String(b,"GBK")//以GBK形式对原字节编码赋值给str
Reader的子类InputStreamReader和Writer的子类OutputStreamWriter提供了字节流和字符流的转换。
InputStreamReader
字节流--->字符流。是字节流通向字符流的桥梁。如果不指定字符集编码,该解码过程将使用平台默认的字符编码,如:GBK。
InputStreamReader isr = new InputStreamReader(new InputStream(),Charset cs);//
适用:适用于在某些应用中文本文件或文本内容初始获取时是字节流的形式。
OutputStreamWriter
字符流--->字节流。可使用指定的 charset 将要写入流中的字符编码成字节。
OutputStreamWriter(OutputStream out, Charset cs)
适用于只接受处理字节流的输入,需要将文本文件或者内容的字符流写入。
NIO是Java新的IO处理机制,它们最大的不同是NIO对流实现了缓冲区机制,不同于传统的IO是从程序员封装实现的,它是从底层操作系统实现的,其效率更高。
以输入字符流为例
FileReader fr = null;
FileWriter fw = null;
try{}catch(){}
finally{
try{
if (null != fr){
fr.close();
}
}catch (IOException e){
throw new RuntimeException("关闭失败");
}
try{
if (null != fw){
fr.close();
}
}catch (IOException e){
throw new RuntimeException("关闭失败");
}
}//finally