Java IO体系

个人觉得可以用“字节流操作类和字符流操作类组成了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. package io;
  2.  
  3. import java.io.FileInputStream;
  4. import java.io.FileNotFoundException;
  5. import java.io.FileOutputStream;
  6. import java.io.FileReader;
  7. import java.io.FileWriter;
  8. import java.io.IOException;
  9.  
  10. public class TestIO {
  11. public static void FileInputStreamTest() throws IOException {
  12. FileInputStream fis = new FileInputStream("tmp2.txt");
  13. byte[] buf = new byte[1024];
  14. int hasRead = 0;
  15.  
  16. //read()返回的是单个字节数据(字节数据可以直接专程int类型),但是read(buf)返回的是读取到的字节数,真正的数据保存在buf中
  17. while ((hasRead = fis.read(buf)) > 0) {
  18. //每次最多将1024个字节转换成字符串,这里tmp2.txt中的字符小于1024,所以一次就读完了
  19. //循环次数 = 文件字符数 除以 buf长度
  20. System.out.println(new String(buf, 0 ,hasRead));
  21. /*
  22. * 将字节强制转换成字符后逐个输出,能实现和上面一样的效果。但是如果源文件是中文的话可能会乱码
  23.  
  24. for (byte b : buf) {
  25. char ch = (char)b;
  26. if (ch != '\r')
  27. System.out.print(ch);
  28. }
  29. */
  30. }
  31. //在finally块里close更安全
  32. fis.close();
  33. }
  34.  
  35. public static void FileReaderTest() throws IOException {
  36.  
  37. try (
  38. // 在try() 中打开的文件, JVM会自动关闭
  39. FileReader fr = new FileReader("tmp2.txt")) {
  40. char[] buf = new char[32];
  41. int hasRead = 0;
  42. // 每个char都占两个字节,每个字符或者汉字都是占2个字节,因此无论buf长度为多少,总是能读取中文字符长度的整数倍,不会乱码
  43. while ((hasRead = fr.read(buf)) > 0) {
  44. // 如果buf的长度大于文件每行的长度,就可以完整输出每行,否则会断行。
  45. // 循环次数 = 文件字符数 除以 buf长度
  46. System.out.println(new String(buf, 0, hasRead));
  47. // 跟上面效果一样
  48. // System.out.println(buf);
  49. }
  50. } catch (IOException ex) {
  51. ex.printStackTrace();
  52. }
  53. }
  54.  
  55. public static void FileOutputStreamTest() throws FileNotFoundException, IOException {
  56. try (
  57. //在try()中打开文件会在结尾自动关闭
  58. FileInputStream fis = new FileInputStream("tmp2.txt");
  59. FileOutputStream fos = new FileOutputStream("tmp3.txt");
  60. ) {
  61. byte[] buf = new byte[4];
  62. int hasRead = 0;
  63. while ((hasRead = fis.read(buf)) > 0) {
  64. //每读取一次就写一次,读多少就写多少
  65. fos.write(buf, 0, hasRead);
  66. }
  67. System.out.println("write success");
  68. } catch (IOException e) {
  69. e.printStackTrace();
  70. }
  71. }
  72.  
  73. public static void FileWriterTest() throws IOException {
  74. try (FileWriter fw = new FileWriter("tmp4.txt")) {
  75. fw.write("天王盖地虎\r\n");
  76. fw.write("宝塔镇河妖\r\n");
  77. } catch (IOException e) {
  78. e.printStackTrace();
  79. }
  80. }
  81. public static void main(String[] args) throws IOException {
  82. //FileInputStreamTest();
  83. //FileReaderTest();
  84. //FileOutputStreamTest();
  85. FileWriterTest();
  86. }
  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. package io;
  2.  
  3. import java.io.BufferedReader;
  4. import java.io.FileInputStream;
  5. import java.io.FileNotFoundException;
  6. import java.io.FileOutputStream;
  7. import java.io.FileReader;
  8. import java.io.IOException;
  9. import java.io.InputStreamReader;
  10. import java.io.PrintStream;
  11. import java.io.PushbackReader;
  12. import java.io.StringReader;
  13. import java.io.StringWriter;
  14.  
  15. public class TestIO {
  16. public static void printStream() throws FileNotFoundException, IOException {
  17. try (
  18. FileOutputStream fos = new FileOutputStream("tmp.txt");
  19. PrintStream ps = new PrintStream(fos)) {
  20. ps.println("普通字符串\n");
  21. //输出对象
  22. ps.println(new TestIO());
  23. } catch (IOException e) {
  24. e.printStackTrace();
  25. }
  26. System.out.println("输出完成");
  27.  
  28. }
  29. public static void stringNode() throws IOException {
  30. String str = "天王盖地虎\n"
  31. + "宝塔镇河妖\n";
  32. char[] buf = new char[32];
  33. int hasRead = 0;
  34. //StringReader将以String字符串为节点读取数据
  35. try (StringReader sr = new StringReader(str)) {
  36. while ((hasRead = sr.read(buf)) > 0) {
  37. System.out.print(new String(buf, 0, hasRead));
  38. }
  39. } catch (IOException e) {
  40. e.printStackTrace();
  41. }
  42.  
  43. //由于String是一个不可变类,因此创建StringWriter时,实际上是以一个StringBuffer作为输出节点
  44. try (StringWriter sw = new StringWriter()) {
  45. sw.write("黑夜给了我黑色的眼睛\n");
  46. sw.write("我却用它寻找光明\n");
  47. //toString()返回sw节点内的数据
  48. System.out.println(sw.toString());
  49. } catch (IOException e) {
  50. e.printStackTrace();
  51. }
  52. }
  53.  
  54. public static void keyIn() throws IOException {
  55. try (
  56. //InputStreamReader是从byte转成char的桥梁
  57. InputStreamReader reader = new InputStreamReader(System.in);
  58. //BufferedReader(Reader in)是char类型输入的包装类
  59. BufferedReader br = new BufferedReader(reader);
  60. ) {
  61. String line = null;
  62. while ((line = br.readLine()) != null) {
  63. if (line.equals("exit")) {
  64. //System.exit(1);
  65. break;
  66. }
  67. System.out.println(line);
  68. }
  69. } catch (IOException e) {
  70. e.printStackTrace();
  71. }
  72. }
  73.  
  74. public static void pushback() throws FileNotFoundException, IOException {
  75. try (PushbackReader pr = new PushbackReader(new FileReader("C:/PROJECT/JavaBasic/PROJECT_JavaBasic/src/io/TestIO.java"),64)) {
  76. char[] buf = new char[32];
  77. String lastContent = "";
  78. int hasRead = 0;
  79. while ((hasRead = pr.read(buf)) > 0) {
  80. String content = new String(buf, 0, hasRead);
  81. int targetIndex = 0;
  82. if ((targetIndex = (lastContent + content).indexOf("targetIndex = (lastContent + content)")) > 0) {
  83. pr.unread((lastContent + content).toCharArray());
  84. if (targetIndex > 32) {
  85. buf = new char[targetIndex];
  86. }
  87. pr.read(buf , 0 , targetIndex);
  88. System.out.println(new String(buf, 0 , targetIndex));
  89. System.exit(0);
  90. } else {
  91. System.out.println(lastContent);
  92. lastContent = content;
  93. }
  94. }
  95. } catch (IOException e) {
  96. e.printStackTrace();
  97. }
  98. }
  99.  
  100. public static void main(String[] args) throws IOException {
  101. printStream();
  102. //stringNode();
  103. //keyIn();
  104. //pushback();
  105. }
  106. }

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

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

JAVA基础知识之IO——Java IO体系及常用类的更多相关文章

  1. Java基础知识二次学习--第六章 常用类

    第六章 常用类   时间:2017年4月26日16:14:49~2017年4月26日16:56:02 章节:06章_01节~06章_06节 视频长度:20:57+1:15+8:44+1:26+11:2 ...

  2. java基础知识回顾之---java String final类普通方法

    辞职了,最近一段时间在找工作,把在大二的时候学习java基础知识回顾下,拿出来跟大家分享,如果有问题,欢迎大家的指正. /*     * 按照面向对象的思想对字符串进行功能分类.     *      ...

  3. java基础知识回顾之java集合类-Properties集合

    /** java.lang.Object   |--java.util.Dictionary<K,V>      |--java.util.Hashtable<Object,Obje ...

  4. java 基础知识一 初识java

    java  基础知识一初识java 1.java语言的特点 (1)简洁有效(2)可移植性(3)面向对象(4)解释型(5)适合分布式计算 2.java的源文件扩展名为.java 编译后的扩展名为.cla ...

  5. java基础知识——网络编程、IO流

    IO流 字节流:处理字节数据的流对象,计算机中最小数据单元就是字节.InputStream OutputStream 字符流:字符编码问题,将字节流和编码表封装成对象就是字符流.Reader Writ ...

  6. JAVA基础知识总结16(IO流)

    IO流:用于处理设备上数据. 流:可以理解数据的流动,就是一个数据流.IO流最终要以对象来体现,对象都存在IO包中. 流也进行分类: 1:输入流(读)和输出流(写). 2:因为处理的数据不同,分为字节 ...

  7. java基础知识回顾之java Socket学习(二)--TCP协议编程

    TCP传输(传输控制协议):TCP协议是一种面向连接的,可靠的字节流服务.当客户端和服务器端彼此交换数据前,必须先在双方之间建立一个TCP连接,之后才能进行数据的传输.它将一台主机发出的字节流无差错的 ...

  8. java基础知识回顾之java Socket学习(一)--UDP协议编程

    UDP传输:面向无连接的协议,不可靠,只是把应用程序传给IP层的数据报包发送出去,不保证发送出去的数据报包能到达目的地.不用再客户端和服务器端建立连接,没有超时重发等机制,传输速度快是它的优点.就像寄 ...

  9. java基础知识回顾之java Thread类学习(十)--线程的状态以及转化使用的方法介绍

       线程的概述:         线程是程序的多个执行路径,执行调度的单位,依托于进程存在.线程不仅可以共享进程的内存,而且还拥有一个属于自己的内存空间,这段内存空间叫做线程栈,是建立线程的时候由系 ...

  10. java基础知识回顾之java Thread类学习(八)--java.util.concurrent.locks(JDK1.5)与synchronized异同讲解

    看API文档介绍几个方法:  JDK1.5中提供了多线程的升级解决方案: 特点: 1.将同步synchronized显示的替换成Lock                    2.接口Conditio ...

随机推荐

  1. Leetcode: Max Sum of Rectangle No Larger Than K

    Given a non-empty 2D matrix matrix and an integer k, find the max sum of a rectangle in the matrix s ...

  2. JQuery下CheckBox全选全不选反选

    <script src="JS/jquery-1.7.1.js"></script> <script type="text/javascri ...

  3. POJ 1811 Prime Test(Miller-Rabin & Pollard-rho素数测试)

    Description Given a big integer number, you are required to find out whether it's a prime number. In ...

  4. Hashing function

    Hashing function (散列函式) 在网页应用中被广泛采用,从数码签署.错误检测.登入验证.到压缩储存空间,由于它的原理比较复杂,很多人把它跟加密函式混淆,对于如何运用hash funct ...

  5. java数组获取最值

    import java.util.Random; /** * 获取最大值最小值 * @author shaobn * */ public class Test2 { public static voi ...

  6. [PHP100]留言板(一)

    [实例]我的留言板 ** 文件结构: conn.php // 数据库配置 add.php // 操作文件 list.php //列表文件 ** 步骤 建立数据库: phpmyadmin: 建立数据库( ...

  7. $.extend,$.fn.extend,$.fn的区别

    jQuery.extend(object) 为jQuery类添加类方法,可以理解为添加静态方法.如: jQuery.extend({ min: function(a, b) { return a &l ...

  8. UISlider控件属性及方法(转)

    初始化一个Slider   UISlider *slider = [[UISlider alloc]initWithFrame:CGRectMake(0, 400,320 , 20)];   访问UI ...

  9. scala伴生对象

    package com.test.scala.test /** * 伴生对象指的是在类中建立一个object */ class AssociatedObject { private var count ...

  10. 给MD5加上salt随机盐值加密算法实现密码安全的php实现

    给MD5加上salt随机盐值加密算法实现密码安全的php实现 如果直接对密码进行散列,那么黑客可以对通过获得这个密码散列值,然后通过查散列值字典(例如MD5密码破解网站),得到某用户的密码.加上sal ...