java IO体系及常见类

一.javaIO体系

 1.基于字节的IO操作


java基于字节的输入操作


java基于字符的输出操作

2.基于字符的IO操作


基于字符的输入操作


基于字符的输出操作

从上图可以看到,整个Java IO体系都是基于字符流(InputStream/OutputStream) 和 字节流(Reader/Writer)作为基类,根据不同的数据载体或功能派生出来的。

二、IO常见类

文件流:FileInputStream/FileOutputStream, FileReader/FileWriter

这四个类是专门操作文件流的,用法高度相似,区别在于前面两个是操作字节流,后面两个是操作字符流。它们都会直接操作文件流,直接与OS底层交互。因此他们也被称为节点流。

注意使用这几个流的对象之后,需要关闭流对象,因为java垃圾回收器不会主动回收。不过在Java7之后,可以在 try() 括号中打开流,最后程序会自动关闭流对象,不再需要显示地close。

下面演示这四个流对象的基本用法:

package 博客;

import java.io.FileInputStream;

import java.io.FileNotFoundException;

import java.io.FileOutputStream;

import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;

public class TestIO {

  public static void FileInputStreamTest() throws IOException {

    FileInputStream fis = new FileInputStream("tmp2.txt");

    byte[] buf = new byte[1024];

    int hasRead = 0;

    // read()返回的是单个字节数据(字节数据可以直接专程int类型),但是read(buf)返回的是读取到的字节数,真正的数据保存在buf中

    while ((hasRead = fis.read(buf)) > 0) {

      // 每次最多将 个字节转换成字符串,这里tmp .txt中的字符小于 ,所以一次就读完了

      // 循环次数 = 文件字符数 除以 buf长度

      System.out.println(new String(buf, 0, hasRead));

      /*

      * 将字节强制转换成字符后逐个输出,能实现和上面一样的效果。但是如果源文件是中文的话可能会乱码

      *

      * for (byte b : buf) { char ch = (char)b; if (ch != '\r') System.out.print(ch);

      * }

      */

    }

    // 在finally块里close更安全

    fis.close();

  }

  public static void FileReaderTest() throws IOException {

    try (

        // 在try() 中打开的文件, JVM会自动关闭

        FileReader fr = new FileReader("tmp .txt")) {

      char[] buf = new char[32];

      int hasRead = 0;

      // 每个char都占两个字节,每个字符或者汉字都是占 个字节,因此无论buf长度为多少,总是能读取中文字符长度的整数倍,不会乱码

      while ((hasRead = fr.read(buf)) > 0) {

        // 如果buf的长度大于文件每行的长度,就可以完整输出每行,否则会断行。

        // 循环次数 = 文件字符数 除以 buf长度

        System.out.println(new String(buf, 0, hasRead));

        // 跟上面效果一样

        // System.out.println(buf);

      }

    } catch (IOException ex) {

      ex.printStackTrace();

    }

  }

  public static void FileOutputStreamTest() throws FileNotFoundException, IOException {

    try (

        // 在try()中打开文件会在结尾自动关闭

        FileInputStream fis = new FileInputStream("tmp .txt");

        FileOutputStream fos = new FileOutputStream("tmp .txt");) {

      byte[] buf = new byte[4];

      int hasRead = 0;

      while ((hasRead = fis.read(buf)) > 0) {

        // 每读取一次就写一次,读多少就写多少

        fos.write(buf, 0, hasRead);

      }

      System.out.println("write success");

    } catch (IOException e) {

      e.printStackTrace();

    }

  }

  public static void FileWriterTest() throws IOException {

    try (FileWriter fw = new FileWriter("tmp .txt")) {

      fw.write("天王盖地虎\r\n");

      fw.write("宝塔镇河妖\r\n");

    } catch (IOException e) {

      e.printStackTrace();

    }

  }

  public static void main(String[] args) throws IOException {

    // FileInputStreamTest();

    // FileReaderTest();

    // FileOutputStreamTest();

    FileWriterTest();

  }

}

包装流:PrintStream/PrintWriter/Scanner

PrintStream可以封装(包装)直接与文件交互的节点流对象OutputStream, 使得编程人员可以忽略设备底层的差异,进行一致的IO操作。因此这种流也称为处理流或者包装流。

PrintWriter除了可以包装字节流OutputStream之外,还能包装字符流Writer

Scanner可以包装键盘输入,方便地将键盘输入的内容转换成我们想要的数据类型。

字符串流:StringReader/StringWriter

这两个操作的是专门操作String字符串的流,其中StringReader能从String中方便地读取数据并保存到char数组,而StringWriter则将字符串类型的数据写入到StringBuffer中(因为String不可写)。

转换流:InputStreamReader/OutputStreamReader

这两个类可以将字节流转换成字符流,被称为字节流与字符流之间的桥梁。我们经常在读取键盘输入(System.in)或网络通信的时候,需要使用这两个类

缓冲流:BufferedReader/BufferedWriter , BufferedInputStream/BufferedOutputStream


    没有经过Buffered处理的IO, 意味着每一次读和写的请求都会由OS底层直接处理,这会导致非常低效的问题。

    经过Buffered处理过的输入流将会从一个buffer内存区域读取数据,本地API只会在buffer空了之后才会被调用(可能一次调用会填充很多数据进buffer)。

    经过Buffered处理过的输出流将会把数据写入到buffer中,本地API只会在buffer满了之后才会被调用。

package 博客;

import java.io.BufferedReader;

import java.io.FileNotFoundException;

import java.io.FileOutputStream;

import java.io.FileReader;

import java.io.IOException;

import java.io.InputStreamReader;

import java.io.PrintStream;

import java.io.PushbackReader;

import java.io.StringReader;

import java.io.StringWriter;

public class TestIO {

  public static void printStream() throws FileNotFoundException, IOException {

    try (FileOutputStream fos = new FileOutputStream("tmp.txt");

        PrintStream ps = new PrintStream(fos)) {

      ps.println("普通字符串\n");

      // 输出对象

      ps.println(new TestIO());

    } catch (IOException e) {

      e.printStackTrace();

    }

    System.out.println("输出完成");

  }

  public static void stringNode() throws IOException {

    String str = "天王盖地虎\n" + "宝塔镇河妖\n";

    char[] buf = new char[32];

    int hasRead = 0;

    // StringReader将以String字符串为节点读取数据

    try (StringReader sr = new StringReader(str)) {

      while ((hasRead = sr.read(buf)) > 0) {

        System.out.print(new String(buf, 0, hasRead));

      }

    } catch (IOException e) {

      e.printStackTrace();

    }

    // 由于String是一个不可变类,因此创建StringWriter时,实际上是以一个StringBuffer作为输出节点

    try (StringWriter sw = new StringWriter()) {

      sw.write("黑夜给了我黑色的眼睛\n");

      sw.write("我却用它寻找光明\n");

      // toString()返回sw节点内的数据

      System.out.println(sw.toString());

    } catch (IOException e) {

      e.printStackTrace();

    }

  }

  public static void keyIn() throws IOException {

    try (

        // InputStreamReader是从byte转成char的桥梁

        InputStreamReader reader = new InputStreamReader(System.in);

        // BufferedReader(Reader in)是char类型输入的包装类

        BufferedReader br = new BufferedReader(reader);) {

      String line = null;

      while ((line = br.readLine()) != null) {

        if (line.equals("exit")) {

          // System.exit(1);

          break;

        }

        System.out.println(line);

      }

    } catch (IOException e) {

      e.printStackTrace();

    }

  }

  public static void pushback() throws FileNotFoundException, IOException {

    try (PushbackReader pr = new PushbackReader(

        new FileReader("C:/PROJECT/JavaBasic/PROJECT_JavaBasic/src/io/TestIO.java"), 64)) {

      char[] buf = new char[32];

      String lastContent = "";

      int hasRead = 0;

      while ((hasRead = pr.read(buf)) > 0) {

        String content = new String(buf, 0, hasRead);

        int targetIndex = 0;

        if ((targetIndex = (lastContent + content)

            .indexOf("targetIndex = (lastContent + content)")) > 0) {

          pr.unread((lastContent + content).toCharArray());

          if (targetIndex > 32) {

            buf = new char[targetIndex];

          }

          pr.read(buf, 0, targetIndex);

          System.out.println(new String(buf, 0, targetIndex));

          System.exit(0);

        } else {

          System.out.println(lastContent);

          lastContent = content;

        }

      }

    } catch (IOException e) {

      e.printStackTrace();

    }

  }

  public static void main(String[] args) throws IOException {

    printStream();

    // stringNode();

    // keyIn();

    // pushback();

  }

}

总结上面几种流的应用场景:

    FileInputStream/FileOutputStream  需要逐个字节处理原始二进制流的时候使用,效率低下

    FileReader/FileWriter 需要组个字符处理的时候使用

    StringReader/StringWriter 需要处理字符串的时候,可以将字符串保存为字符数组

    PrintStream/PrintWriter 用来包装FileOutputStream 对象,方便直接将String字符串写入文件 

    Scanner 用来包装System.in流,很方便地将输入的String字符串转换成需要的数据类型

    InputStreamReader/OutputStreamReader ,  字节和字符的转换桥梁,在网络通信或者处理键盘输入的时候用

    BufferedReader/BufferedWriter , BufferedInputStream/BufferedOutputStream , 缓冲流用来包装字节流后者字符流,提升IO性能,BufferedReader还可以方便地读取一行,简化编程。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 207,113评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,644评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,340评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,449评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,445评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,166评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,442评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,105评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,601评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,066评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,161评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,792评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,351评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,352评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,584评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,618评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,916评论 2 344

推荐阅读更多精彩内容

  • tags:io categories:总结 date: 2017-03-28 22:49:50 不仅仅在JAVA领...
    行径行阅读 2,168评论 0 3
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,194评论 11 349
  • 恰到好处的喜欢最舒服,就好比 你随口说了句肚子好饿 我就跑到楼下给你买吃的。你说你累了 我就让你靠在我旁边。你说你...
    嘟嘟咚咚呛阅读 468评论 0 0
  • 为什么说微习惯,这个肯定行呢?是因为我已经在坚持,并且做到了。 我来谈谈自己的体会: 我坚持的第一个微习惯是参加好...
    华枝春满5339阅读 291评论 4 4
  • 2017.4.19原來我不願意離開是因為不甘心。看完這篇文章,狠狠地豪哭一番。原來是不甘心呀!不甘心、不甘心、不甘...
    禪心園文創設計工作室阅读 689评论 3 12