更多Android高级架构进阶视频学习请点击:https://space.bilibili.com/474380680

个人觉得可以用“字节流操作类和字符流操作类组成了Java IO体系”来高度概括Java IO体系。

借用几张网络图片来说明(图片来自 http://blog.csdn.net/zhangerqing/article/details/8466532 )

** 基于字节的IO操作**

|

 

|

 
 

|
基于字符的IO操作

|

 

|

 

|

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

IO常用类

  • 文件流:FileInputStream/FileOutputStream, FileReader/FileWriter

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

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

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

  1. 1 package io;
  2. 2
  3. 3 import java.io.FileInputStream;
  4. 4 import java.io.FileNotFoundException;
  5. 5 import java.io.FileOutputStream;
  6. 6 import java.io.FileReader;
  7. 7 import java.io.FileWriter;
  8. 8 import java.io.IOException;
  9. 9
  10. 10 public class TestIO {
  11. 11 public static void FileInputStreamTest() throws IOException {
  12. 12 FileInputStream fis = new FileInputStream("tmp2.txt");
  13. 13 byte[] buf = new byte[1024];
  14. 14 int hasRead = 0;
  15. 15
  16. 16 //read()返回的是单个字节数据(字节数据可以直接专程int类型),但是read(buf)返回的是读取到的字节数,真正的数据保存在buf中
  17. 17 while ((hasRead = fis.read(buf)) > 0) {
  18. 18 //每次最多将1024个字节转换成字符串,这里tmp2.txt中的字符小于1024,所以一次就读完了
  19. 19 //循环次数 = 文件字符数 除以 buf长度
  20. 20 System.out.println(new String(buf, 0 ,hasRead));
  21. 21 /*
  22. 22 * 将字节强制转换成字符后逐个输出,能实现和上面一样的效果。但是如果源文件是中文的话可能会乱码
  23. 23
  24. 24 for (byte b : buf) {
  25. 25 char ch = (char)b;
  26. 26 if (ch != '\r')
  27. 27 System.out.print(ch);
  28. 28 }
  29. 29 */
  30. 30 }
  31. 31 //在finally块里close更安全
  32. 32 fis.close();
  33. 33 }
  34. 34
  35. 35 public static void FileReaderTest() throws IOException {
  36. 36
  37. 37 try (
  38. 38 // 在try() 中打开的文件, JVM会自动关闭
  39. 39 FileReader fr = new FileReader("tmp2.txt")) {
  40. 40 char[] buf = new char[32];
  41. 41 int hasRead = 0;
  42. 42 // 每个char都占两个字节,每个字符或者汉字都是占2个字节,因此无论buf长度为多少,总是能读取中文字符长度的整数倍,不会乱码
  43. 43 while ((hasRead = fr.read(buf)) > 0) {
  44. 44 // 如果buf的长度大于文件每行的长度,就可以完整输出每行,否则会断行。
  45. 45 // 循环次数 = 文件字符数 除以 buf长度
  46. 46 System.out.println(new String(buf, 0, hasRead));
  47. 47 // 跟上面效果一样
  48. 48 // System.out.println(buf);
  49. 49 }
  50. 50 } catch (IOException ex) {
  51. 51 ex.printStackTrace();
  52. 52 }
  53. 53 }
  54. 54
  55. 55 public static void FileOutputStreamTest() throws FileNotFoundException, IOException {
  56. 56 try (
  57. 57 //在try()中打开文件会在结尾自动关闭
  58. 58 FileInputStream fis = new FileInputStream("tmp2.txt");
  59. 59 FileOutputStream fos = new FileOutputStream("tmp3.txt");
  60. 60 ) {
  61. 61 byte[] buf = new byte[4];
  62. 62 int hasRead = 0;
  63. 63 while ((hasRead = fis.read(buf)) > 0) {
  64. 64 //每读取一次就写一次,读多少就写多少
  65. 65 fos.write(buf, 0, hasRead);
  66. 66 }
  67. 67 System.out.println("write success");
  68. 68 } catch (IOException e) {
  69. 69 e.printStackTrace();
  70. 70 }
  71. 71 }
  72. 72
  73. 73 public static void FileWriterTest() throws IOException {
  74. 74 try (FileWriter fw = new FileWriter("tmp4.txt")) {
  75. 75 fw.write("天王盖地虎\r\n");
  76. 76 fw.write("宝塔镇河妖\r\n");
  77. 77 } catch (IOException e) {
  78. 78 e.printStackTrace();
  79. 79 }
  80. 80 }
  81. 81 public static void main(String[] args) throws IOException {
  82. 82 //FileInputStreamTest();
  83. 83 //FileReaderTest();
  84. 84 //FileOutputStreamTest();
  85. 85 FileWriterTest();
  86. 86 }
  87. 87 }

  • 包装流: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

Oracle官方的描述:

Most of the examples we've seen so far use unbuffered I/O. This means each read or write request is handled directly by the underlying OS. This can make a program much less efficient.

Buffered input streams read data from a memory area known as a buffer; the native input API is called only when the buffer is empty. Similarly, buffered output streams write data to a buffer, and the native output API is called only when the buffer is full.

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

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

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

BufferedReader/BufferedWriter可以将字符流(Reader)包装成缓冲流,这是最常见用的做法。

另外,BufferedReader提供一个readLine()可以方便地读取一行,而FileInputStream和FileReader只能读取一个字节或者一个字符,

因此BufferedReader也被称为行读取器

下面演示上面提到的常见类,

  1. 1 package io;
  2. 2
  3. 3 import java.io.BufferedReader;
  4. 4 import java.io.FileInputStream;
  5. 5 import java.io.FileNotFoundException;
  6. 6 import java.io.FileOutputStream;
  7. 7 import java.io.FileReader;
  8. 8 import java.io.IOException;
  9. 9 import java.io.InputStreamReader;
  10. 10 import java.io.PrintStream;
  11. 11 import java.io.PushbackReader;
  12. 12 import java.io.StringReader;
  13. 13 import java.io.StringWriter;
  14. 14
  15. 15 public class TestIO {
  16. 16 public static void printStream() throws FileNotFoundException, IOException {
  17. 17 try (
  18. 18 FileOutputStream fos = new FileOutputStream("tmp.txt");
  19. 19 PrintStream ps = new PrintStream(fos)) {
  20. 20 ps.println("普通字符串\n");
  21. 21 //输出对象
  22. 22 ps.println(new TestIO());
  23. 23 } catch (IOException e) {
  24. 24 e.printStackTrace();
  25. 25 }
  26. 26 System.out.println("输出完成");
  27. 27
  28. 28 }
  29. 29 public static void stringNode() throws IOException {
  30. 30 String str = "天王盖地虎\n"
  31. 31 + "宝塔镇河妖\n";
  32. 32 char[] buf = new char[32];
  33. 33 int hasRead = 0;
  34. 34 //StringReader将以String字符串为节点读取数据
  35. 35 try (StringReader sr = new StringReader(str)) {
  36. 36 while ((hasRead = sr.read(buf)) > 0) {
  37. 37 System.out.print(new String(buf, 0, hasRead));
  38. 38 }
  39. 39 } catch (IOException e) {
  40. 40 e.printStackTrace();
  41. 41 }
  42. 42
  43. 43 //由于String是一个不可变类,因此创建StringWriter时,实际上是以一个StringBuffer作为输出节点
  44. 44 try (StringWriter sw = new StringWriter()) {
  45. 45 sw.write("黑夜给了我黑色的眼睛\n");
  46. 46 sw.write("我却用它寻找光明\n");
  47. 47 //toString()返回sw节点内的数据
  48. 48 System.out.println(sw.toString());
  49. 49 } catch (IOException e) {
  50. 50 e.printStackTrace();
  51. 51 }
  52. 52 }
  53. 53
  54. 54 public static void keyIn() throws IOException {
  55. 55 try (
  56. 56 //InputStreamReader是从byte转成char的桥梁
  57. 57 InputStreamReader reader = new InputStreamReader(System.in);
  58. 58 //BufferedReader(Reader in)是char类型输入的包装类
  59. 59 BufferedReader br = new BufferedReader(reader);
  60. 60 ) {
  61. 61 String line = null;
  62. 62 while ((line = br.readLine()) != null) {
  63. 63 if (line.equals("exit")) {
  64. 64 //System.exit(1);
  65. 65 break;
  66. 66 }
  67. 67 System.out.println(line);
  68. 68 }
  69. 69 } catch (IOException e) {
  70. 70 e.printStackTrace();
  71. 71 }
  72. 72 }
  73. 73
  74. 74 public static void pushback() throws FileNotFoundException, IOException {
  75. 75 try (PushbackReader pr = new PushbackReader(new FileReader("C:/PROJECT/JavaBasic/PROJECT_JavaBasic/src/io/TestIO.java"),64)) {
  76. 76 char[] buf = new char[32];
  77. 77 String lastContent = "";
  78. 78 int hasRead = 0;
  79. 79 while ((hasRead = pr.read(buf)) > 0) {
  80. 80 String content = new String(buf, 0, hasRead);
  81. 81 int targetIndex = 0;
  82. 82 if ((targetIndex = (lastContent + content).indexOf("targetIndex = (lastContent + content)")) > 0) {
  83. 83 pr.unread((lastContent + content).toCharArray());
  84. 84 if (targetIndex > 32) {
  85. 85 buf = new char[targetIndex];
  86. 86 }
  87. 87 pr.read(buf , 0 , targetIndex);
  88. 88 System.out.println(new String(buf, 0 , targetIndex));
  89. 89 System.exit(0);
  90. 90 } else {
  91. 91 System.out.println(lastContent);
  92. 92 lastContent = content;
  93. 93 }
  94. 94 }
  95. 95 } catch (IOException e) {
  96. 96 e.printStackTrace();
  97. 97 }
  98. 98 }
  99. 99
  100. 100 public static void main(String[] args) throws IOException {
  101. 101 printStream();
  102. 102 //stringNode();
  103. 103 //keyIn();
  104. 104 //pushback();
  105. 105 }
  106. 106 }

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

  • FileInputStream/FileOutputStream 需要逐个字节处理原始二进制流的时候使用,效率低下
  • FileReader/FileWriter 需要组个字符处理的时候使用
  • StringReader/StringWriter 需要处理字符串的时候,可以将字符串保存为字符数组
  • PrintStream/PrintWriter 用来包装FileOutputStream 对象,方便直接将String字符串写入文件
  • Scanner 用来包装System.in流,很方便地将输入的String字符串转换成需要的数据类型
  • InputStreamReader/OutputStreamReader , 字节和字符的转换桥梁,在网络通信或者处理键盘输入的时候用
  • BufferedReader/BufferedWriter , BufferedInputStream/BufferedOutputStream , 缓冲流用来包装字节流后者字符流,提升IO性能,BufferedReader还可以方便地读取一行,简化编程。

更多Android高级架构进阶视频学习请点击:https://space.bilibili.com/474380680

原文转载https://www.cnblogs.com/fysola/p/6123947.html

高效IO之Java IO体系(一)的更多相关文章

  1. hive运行query语句时提示错误:org.apache.hadoop.ipc.RemoteException: java.io.IOException: java.io.IOException:

    hive> select product_id, track_time from trackinfo limit 5; Total MapReduce jobs = 1 Launching Jo ...

  2. ava.io.InputStream & java.io.FileInputStream

    java.io.InputStream & java.io.FileInputStream java.io.InputStream,这个抽象类是表示字节输入流的超类,这个抽象类的共性的方法有: ...

  3. java.io.OutputStream & java.io.FileOutputStream

    java.io.OutputStream & java.io.FileOutputStream 1.Java.io.OutputStream(字节输出流) 字节输出流,这是一个抽象类,是表示输 ...

  4. 缓冲字符流 java.io.BufferedWriter ,java.io.BufferedReader,缓冲字符输出流:PrintWriter

    package seday07; import java.io.IOException;import java.io.PrintWriter; /*** @author xingsir * 缓冲字符流 ...

  5. 转换流读写操作 java.io.OutputStreamWriter ,java.io.InputStreamReader

    package seday07; import java.io.FileOutputStream;import java.io.IOException;import java.io.OutputStr ...

  6. 对象流,它们是一对高级流,负责即将java对象与字节之间在读写的过程中进行转换。 * java.io.ObjectOutputStream * java.io.ObjectInputStream

    package seday06; import java.io.Serializable;import java.util.Arrays; /** * @author xingsir * 使用当前类来 ...

  7. java.io.IOException: java.io.FileNotFoundException: /tmp/tomcat.2457258178644046891.8080/work/Tomcat/localhost/innovate-admin/C:/up/154884318438733213952/sys-error.log (没有那个文件或目录)

    环境: Ubuntu18 vue+elementUI 实现文件的上传 报错信息: MultipartFile.transferTo(dest) 报 FileNotFoundException java ...

  8. java.io.EOFException java.io.ObjectInputStream$PeekInputStream.readFully 错误

    Tomcat 启动时 java.io.EOFException at java.io.ObjectInputStream$PeekInputStream.readFully 错误 这 个错误 碰到好几 ...

  9. 一头扎进 Java IO中-------java IO文件

    Java IO: 文件 在Java应用程序中,文件是一种常用的数据源或者存储数据的媒介.所以这一小节将会对Java中文件的使用做一个简短的概述.这篇文章不会对每一个技术细节都做出解释,而是会针对文件存 ...

随机推荐

  1. Redis Sentinel用法

    1 Redis Sentinel 1.1 哨兵的作用 1. 监控:监控主从是否正常 2. 通知:出现问题时,可以通知相关人员 3. 故障迁移:自动主从切换 4. 统一的配置管理:连接者询问sentin ...

  2. Java DOM解析器 - 查询XML文档

    这是需要我们查询的输入XML文件: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 <?xml version="1.0"?> ...

  3. Struct和Union的sizeof计算

    struct 结构体的大小不是简单的成员相加,要考虑存储空间的字节对齐 1.空结构体的大小为1 2.含有static的结构体在计算大小时不算上static变量,因为static存储在全局数据空间,而s ...

  4. leetcode.矩阵.240搜索二维矩阵II-Java

    1. 具体题目 编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target.该矩阵具有以下特性:每行的元素从左到右升序排列:每列的元素从上到下升序排列. 示例: 现有矩阵 ...

  5. 分布式-技术专区-Redis和MySQL缓存一致性问题

    1.Redis 缓存和 MySQL 数据如何实现一致性 需求起因 缓存和数据库一致性解决方案 在高并发的业务场景下,数据库大多数情况都是用户并发访问最薄弱的环节.所以,就需要使用redis做一个缓冲操 ...

  6. ubuntu安装goland

    安装goland 首先下载goland https://www.jetbrains.com/zh/go/specials/go/go.html?utm_source=baidu&utm_med ...

  7. 《构建之法》IT行业的创新 读书笔记 WEEK 5

    本周选读邹欣老师的<构建之法>第16章——IT行业的创新. 邹欣老师将本章话题分成五个部分来阐述:创新的迷思.创新的时机.创新的招数.魔方的创新.创新和作坊,博主认为时机和招数这两个部分在 ...

  8. 使用PL/SQL连接oracle数据库,并将数据进行导出备份和导入恢复

    使用PL/SQL连接oracle数据库,并将数据进行导出备份和导入恢复 这种操作百度一搜一大片,今天整理以前做的项目时自己备份了一下数据库,试着将数据进行导出备份和导入恢复了一下:下面是操作过程: 1 ...

  9. Ansible的roles标准化与Jenkins持续集成(三)

    Ansible的roles标准化与Jenkins持续集成(三) 链接:https://pan.baidu.com/s/1A3Iq3gGkGS27L_Gt37_I0g 提取码:ncy2 复制这段内容后打 ...

  10. grep 正则2

    基本正则表达式所定义的元字符 元字符 作用 例子 例子说明 ^ 行首定位符 ^ty 匹配"t"开头,后面紧跟一个"y"的字符串 $ 行尾定位符 txt$ 匹配以 ...