一、什么是流?

流就是字节序列的抽象概念,能被连续读取数据的数据源和能被连续写入数据的接收端就是流,流机制是Java及C++中的一个重要机制,通过流我们可以自由地控制文件、内存、IO设备等数据的流向。而IO流就是用于处理设备上的数据,如:硬盘、内存、键盘录入等。IO流根据处理类型的不同可分为字节流和字符流,根据流向的不同可分为输入流和输出流。

二、字节流和字符流的区别:

字符流,因为文件编码的不同,就有了对字符进行高效操作的字符流对象,它的原理就是基于字节流读取字节时去查了指定的码表。它和字节流的区别有两点:1.在读取数据的时候,字节流读到一个字节就返回一个字节,字符流使用了字节流读到一个或多个字节(一个中文对应的字节数是两个,在UTF-8码表中是3个字节)时,先去查指定的编码表,再将查到的字符返回;2.字节流可以处理所有类型的数据,如jpg、avi、mp3、wav等等,而字符流只能处理字符数据。所以可以根据处理的文件不同考虑使用字节流还是字符流,如果是纯文本数据可以优先考虑字符流,否则使用字节流。

三、IO体系,所具备的基本功能就是读和写:

1.字符流

|-- Reader(读)

|-- Writer(写)

Reader

|--InputStreamReader

|--FileReader:用于处理文件的字符读取流对象

Writer

|--OutputStreamWriter

|--FileWriter:用于处理文件的字符写入流对象

其实很容易就可以看出来,IO体系中的子类名后缀绝大部分是父类名称,而前缀则是体现子类特有功能的名称。

Reader中常见的方法:

|--int read()

读取一个字符,并返回读到的这个字符,读到流的末尾则返回-1。

|--int read(char[])

将读到的字符存入指定的数组中,返回的是读到的字符个数,读到流的末尾则返回-1。

|--close()

进行资源的释放。

FileReader除了自己的构造函数外没有特有的方法:

|--用于读取文本文件的流对象。

|--用于关联文本文件。

|--构造函数FileReader(String fileName)

在读取流对象初始化时,必须要指定一个被读取的文件,

如果该文件不存在则会发生FileNotFoundException异常。

Writer中常见的方法:

|--write()

将一个字符写入到流中。

|--write(char[])

将一个字符数组写入到流中。

|--writer(String)

将一个字符写入到流中。

|--flush()

刷新流,将流中的数据刷新到目的地中,流还存在。

|--close()

关闭资源,在关闭钱会先调用flush(), 刷新流中的数据到目的地。

FileWriter,除了自己的构造函数外没有特有的方法:

|--该类的特点

|--用于处理文本文件

|--没有默认的编码表

|--有临时缓冲

|--构造函数,在写入流对象初始化时,必须要有一个存储数据的目的地。

|--FileWriter(String fileName),该构造器是干什么用的呢?

|--调用系统资源

|--在指定位置创建一个文件,如果该文件已经存在则被覆盖。

|--FileWriter(String filename,Boolean append),这构造器的作用是?

当传入的boolean类型的值为true时,会在指定文件末尾处进行数据的续写。

清单1,将文本数据保存到文件中代码

  1. private static void test1(){
  2.  
  3. FileWriter fw=null;
  4.  
  5. try {
  6.  
  7. //初始化FileWriter对象,指定文件名已经存储路径
  8.  
  9. fw=new FileWriter("D:/test.txt");
  10.  
  11. fw.write("将字符串写入流");
  12.  
  13. //将流中的数据刷新到目的地,流还在
  14.  
  15. fw.flush();
  16.  
  17. fw.write("将字符串写入流");
  18.  
  19. } catch (IOException e) {
  20.  
  21. e.printStackTrace();
  22.  
  23. }finally{
  24.  
  25. if(fw!=null){
  26.  
  27. try {
  28.  
  29. fw.close();
  30.  
  31. } catch (IOException e1) {
  32.  
  33. e1.printStackTrace();
  34.  
  35. }
  36.  
  37. }
  38.  
  39. }
  40.  
  41. }

清单2,读取一个已有文本文件,并将文本内容打印出来代码

  1. private static void test2(){
  2.  
  3. FileReader fr=null;
  4.  
  5. try {
  6.  
  7. //初始化FileReader对象,指定文件路径
  8.  
  9. fr=new FileReader("D:/test.txt");
  10.  
  11. int ch=0;
  12.  
  13. while((ch=fr.read())!=-1){
  14.  
  15. //每次读取一个字符,直到读到末尾-1为止
  16.  
  17. System.out.println((char)ch);
  18.  
  19. }
  20.  
  21. } catch (IOException e) {
  22.  
  23. e.printStackTrace();
  24.  
  25. }finally{
  26.  
  27. if(fr!=null){
  28.  
  29. try {
  30.  
  31. fr.close();
  32.  
  33. } catch (IOException e1) {
  34.  
  35. e1.printStackTrace();
  36.  
  37. }
  38.  
  39. }
  40. }
  41.  
  42. }

这样每读到一个字符就打印出来,效率很不高,能不能按指定大小读取完后再打印出来呢?答案是当然可以的。

清单3,读取一个已有文本文件,读完1kb再将其读到的内容打印出来代码

  1. private static void test3(){
  2.  
  3. FileReader fr=null;
  4.  
  5. try {
  6.  
  7. //初始化FileReader对象,指定文件路径
  8.  
  9. fr=new FileReader("D:/test.txt");
  10.  
  11. char[] buf=new char[1024];
  12.  
  13. int len=0;
  14.  
  15. while((len=fr.read(buf))!=-1){
  16.  
  17. //每次读取1kb大小的字符,直到读到末尾-1为止
  18.  
  19. System.out.println(new String(buf,0,len));
  20.  
  21. }
  22.  
  23. } catch (IOException e) {
  24.  
  25. e.printStackTrace();
  26.  
  27. }finally{
  28.  
  29. if(fr!=null){
  30.  
  31. try {
  32.  
  33. fr.close();
  34.  
  35. } catch (IOException e1) {
  36.  
  37. e1.printStackTrace();
  38.  
  39. }
  40.  
  41. }
  42.  
  43. }
  44. }

字符流的缓冲区:

|--缓冲区的出现提高了对流的操作效率。

原理:其实就是将数组进行封装。

|--对应的对象

|--BufferedWriter

特有方法newLine(),跨平台的换行符。

|--BufferedReader

特有方法readLine(),一次读一行,到行标记时,将行标记

之前的字符数据作为字符串返回,读到末尾返回null。

|--说明

在使用缓冲区对象时,要明确,缓冲的存在是为了增强流

的功能而存在,所以在建立缓冲区对象时,要先有流对象

存在。其实缓冲区内部就是在使用流对象的方法,只不过

加入了数组对数据进行了临时存储,为了提高操作数据的

效率。

|--代码上的体现

|--写入缓冲区对象

根据前面所说的建立缓冲区时要先有流对象,并将其作为参数传递给缓冲区的构造函数

  1. BufferedWriter bufw=new BufferedWriter(new FileWriter(“test.txt”));
  2.  
  3. bufw.write(“将数据写入缓冲区”);
  4.  
  5. bufw.flush();//将缓冲区的数据刷新到目的地
  6.  
  7. bufw.close();//其实关闭的是被包装在内部的流对象

|--读取缓冲区对象

  1. BufferedReader bufr=new BufferedReader(new FileReader(“test.txt”));
  2.  
  3. String line=null;
  4.  
  5. while((line=bufr.readLine())!=null){
  6.  
  7. //每次读取一行,取出的数据不包含回车符
  8.  
  9. system.out.println(line);
  10.  
  11. }
  12.  
  13. bufr.close();

清单4,使用缓冲区对文本文件进行拷贝代码

  1. private static void test4(){
  2.  
  3. BufferedReader bufr=null;
  4.  
  5. BufferedWriter bufw=null;
  6.  
  7. try {
  8.  
  9. bufr=new BufferedReader(new FileReader("D:/a.txt"));
  10.  
  11. bufw=new BufferedWriter(new FileWriter("D:/b.txt"));
  12.  
  13. String line=null;
  14.  
  15. while((line=bufr.readLine())!=null){
  16.  
  17. bufw.write(line);//每次将一行写入缓冲区
  18.  
  19. bufw.flush();//刷新到目的地
  20.  
  21. }
  22.  
  23. } catch (IOException e) {
  24.  
  25. e.printStackTrace();
  26.  
  27. }finally{
  28.  
  29. try {
  30.  
  31. if(bufw!=null){
  32.  
  33. bufw.close();
  34.  
  35. }
  36.  
  37. if(bufr!=null){
  38.  
  39. bufr.close();
  40.  
  41. }
  42.  
  43. } catch (IOException e1) {
  44.  
  45. e1.printStackTrace();
  46.  
  47. }
  48.  
  49. }
  50.  
  51. }

仔细看可以发现,程序里面的FileReader对象和FileWriter对象直接new出来且没有调用close(),因为缓冲对象调用了这两个方法,前面说了,缓冲对象调用的flush()和close()其实就是关闭被包装在其内部的流对象。关闭流的先后顺序也要注意,如果流之间有依赖关系,则被依赖的流要后关闭。readLine()方法原理:其实缓冲区中的该方法,用的还是与缓冲区关联的流对象的read方法,只不过,每一次读到一个字符先不进行具体操作,先进行临时存储,当读到回车标记时,将临时容器中存储的数据一次性返回。我们可以根据这个原理来自己编写一个缓冲区对象。

清单5,编写一个自己的bufferedreader代码

  1. public class MyBufferedReader {
  2.  
  3. private Reader reader;
  4.  
  5. public MyBufferedReader(Reader reader){
  6.  
  7. this.reader=reader;
  8.  
  9. }
  10.  
  11. public String readLine() throws IOException{
  12.  
  13. StringBuilder sb=new StringBuilder();
  14.  
  15. int ch=0;
  16.  
  17. while((ch=reader.read())!=-1){
  18.  
  19. if(ch=='\r'){//空格则继续
  20.  
  21. continue;
  22.  
  23. }else if(ch=='\n'){//每次返回一行
  24.  
  25. return sb.toString();
  26.  
  27. }else{
  28.  
  29. sb.append((char)ch);
  30.  
  31. }
  32.  
  33. }
  34.  
  35. return sb.toString();
  36. }
  37. public void close() throws IOException{
  38.  
  39. //缓冲对象的关闭方法其实就是调用流本身的close()
  40.  
  41. reader.close();
  42.  
  43. }
  44.  
  45. }

测试时把清单4的BufferedReader对象替换成MyBufferedReader对象即可。

清单6,测试mybufferedreader代码

  1. private static void test4(){
  2.  
  3. MyBufferedReader bufr=null;
  4.  
  5. BufferedWriter bufw=null;
  6.  
  7. try {
  8.  
  9. bufr=new MyBufferedReader(new FileReader("D:/a.txt"));
  10.  
  11. bufw=new BufferedWriter(new FileWriter("D:/b.txt"));
  12.  
  13. String line=null;
  14.  
  15. while((line=bufr.readLine())!=null){
  16.  
  17. bufw.write(line);//每次将一行写入缓冲区
  18.  
  19. bufw.flush();//刷新到目的地
  20.  
  21. }
  22.  
  23. } catch (IOException e) {
  24.  
  25. e.printStackTrace();
  26.  
  27. }finally{
  28.  
  29. try {
  30.  
  31. if(bufw!=null){
  32.  
  33. bufw.close();
  34.  
  35. }
  36.  
  37. if(bufr!=null){
  38.  
  39. bufr.close();
  40.  
  41. }
  42.  
  43. } catch (IOException e1) {
  44.  
  45. e1.printStackTrace();
  46.  
  47. }
  48.  
  49. }
  50.  
  51. }

其实我们自己写的这个缓存对象就是对Reader对象进行了功能的增强,Reader对象每次只能返回一个字符,而增强了功能之后该类就可以每次返回一行字符,也就是设计模式中所说的装饰模式

Java之IO流学习总结【上】的更多相关文章

  1. java的Io流学习

    Java中io流的学习(一)File:https://blog.csdn.net/qq_41061437/article/details/81672859 Java中io流的学习(二)FileInpu ...

  2. Java之IO流学习总结

    流:可以理解为数据的流动,就是一个数据流,IO流最终要以对象来体现 流的分类:     按照流的方向:输入流和输出流  (输入流只能进行读操作,输出流只能进行写操作)     按照处理数据的不同:字节 ...

  3. java中io流实现文件上传下载

    新建io.jsp <%@ page language="java" contentType="text/html; charset=UTF-8" page ...

  4. Java之IO流学习总结【下】

    2.字节流 |-- InputStream(读) |-- OutputStream(写) 由于字节是二进制数据,所以字节流可以操作任何类型的数据,值得注意的是字符流使用的是字符数组char[]而字节流 ...

  5. JAVA.IO流学习笔记

    一.java.io 的描述 通过数据流.序列化和文件系统提供系统输入和输出.IO流用来处理设备之间的数据传输 二.流 流是一个很形象的概念,当程序需要读取数据的时候,就会开启一个通向数据源的流,这个数 ...

  6. Java IO流学习总结三:缓冲流-BufferedInputStream、BufferedOutputStream

    Java IO流学习总结三:缓冲流-BufferedInputStream.BufferedOutputStream 转载请标明出处:http://blog.csdn.net/zhaoyanjun6/ ...

  7. Java IO流学习总结八:Commons IO 2.5-IOUtils

    Java IO流学习总结八:Commons IO 2.5-IOUtils 转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/550519 ...

  8. Java IO流学习总结(1)

    Java IO流学习总结 Java流操作有关的类或接口: Java流类图结构: 流的概念和作用 流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象.即数据在两设备间的传输称为流,流的本 ...

  9. (转载) java:IO流学习小结

    今天刚刚看完Java的io流操作,把主要的脉络看了一遍,不能保证以后使用时都能得心应手,但是最起码用到时知道有这么一个功能可以实现,下面对学习进行一下简单的总结: IO流主要用于硬板.内存.键盘等处理 ...

随机推荐

  1. HTML5 使用FileReader实现调用相册、拍照功能

    HTML5定义了FileReader作为文件API的重要成员用于读取文件,根据W3C的定义,FileReader接口提供了读取文件的方法和包含读取结果的事件模型. FileReader的使用方式非常简 ...

  2. vue2.0 配置sass

    一.配置sass依赖 npm install node-sass --save-dev npm install sass-loader --save-dev 二.打开build文件夹下的webpack ...

  3. window64 PHP ffmpeg详解简单上手 音频amr转mp3

    从网上找了一大堆关于window 64 ffmpeg的信息,都是又长又不关键,让人难消化. 我只要简单的amr转MP3格式而已. 终于搞明白.自己总结了下! 希望能帮助到喜欢言简意赅,一眼上手的同学. ...

  4. golang动态加载原生代码思路

    golang动态加载原生代码思路(非plugin,非so文件.使用mmap形式运行机器码,可释放) 1.用go tool objdump,可以看到任意函数的机器码.汇编指令.偏移.(go源码下面有一个 ...

  5. 2017 年的 人生 hard 模式终于结束了,2018年回归初心(二)

    今天周末, 深圳的天气简直好的不像话.好了,我们继续之前的话题往下聊. >>>猎头 : 关于猎头这个行业,以笔者的感觉来说 一般你工作年限未超过三年的话,你是很难遇到猎头来推送你的简 ...

  6. django 项目中遇到的问题(持续更新中)

    问题1:in include 'provide the namespace argument to include() instead 描述:在最外层的urls.py 添加项目的urls后报错,错误显 ...

  7. web中的简单全选反选

    <html> <body> <table> <tr> <th><input type="checkbox" onc ...

  8. python 浅析IO 模型

    协程:遇到IO操作就切换,但是什么时候切回去呢?怎么确定IO操作? 很多程序员可能会考虑使用"线程池"或"连接池"."线程池"旨在减少创建和 ...

  9. mysql一些函数的记录

    今天瞎搞还是弄不出报名程序,偶尔记住了几个mysql函数 mysql_connect()连接数据库 mysql_error()输出文本报错 mysql_select_db()函数设置活动的MySQL  ...

  10. HDU4355-Party All the Time-三分

    Party All the Time Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Other ...