`

JAVA基础知识之IO——Java IO体系及常用类

    博客分类:
  • java
 
阅读更多

转自:http://www.cnblogs.com/fysola/p/6123947.html

 

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核心知识点整理——java自学资料.rar

    这份"JAVA核心知识点整理——java自学资料.rar"包含了一份详细的Java学习资源,旨在帮助已经具备一定Java基础知识的开发者进一步提升技能,深入理解Java的底层机制。以下是这份资料可能涵盖的一些核心知识点和学习...

    超详细的Java复习知识点2019——个人笔记.zip

    这份名为"超详细的Java复习知识点2019——个人笔记"的文档,旨在为初学者提供一个全面且深入理解Java基础知识的指南。笔记内容可能涵盖了以下几个关键领域: 1. **Java语法基础**:包括变量、数据类型、运算符、...

    Java图书管理系统(IO流版)(csdn)————程序.pdf

    本文总结了Java图书管理系统(IO流版)的主要知识点,涵盖了Java基础语法、流程控制、面向对象思想、封装、继承、多态、接口、异常、集合、IO流等多方面的知识。 一、Java基础语法 * 掌握Java的基本语法,包括变量...

    java基础——————试题库

    这个“java_test”文件很可能是包含了一系列测试题目的集合,涵盖了以上提到的所有Java基础知识点,通过练习这些题目,你可以检验自己的学习成果,发现并弥补知识盲点,进一步巩固Java基础。在学习过程中,不断实践...

    JAVA学习笔记————————

    1. **JAVA基础**:这部分包括JAVA语法的基础知识,如数据类型(基本类型与引用类型)、变量、运算符、流程控制(条件语句、循环语句)、方法的定义和调用等。此外,还会涉及类和对象的概念,这是面向对象编程的核心...

    Java基础知识——Java入门与加深

    本文将深入探讨Java的基础知识,旨在帮助初学者快速入门并进一步加深对Java的理解。 首先,我们来了解一下Java的核心概念。Java是基于C++语言发展而来的,但去除了指针等复杂特性,使代码更加安全。它的“Write ...

    尚硅谷Java基础实战——Bank项目下载地址

    这个项目以银行系统为背景,涵盖了诸多Java基础知识点,包括类与对象、数据类型、控制结构、异常处理、集合框架、IO流等。 1. 类与对象:在Bank项目中,你会遇到各种类,如Bank、Account、Customer等,这些都是模拟...

    115个Java面试题和答案——终极列表

    1. **Java基础** - **数据类型**:Java分为基本数据类型(如int, double, boolean等)和引用数据类型(类、接口、数组)。理解它们的区别和用法至关重要。 - **变量**:变量是存储数据的容器,分为实例变量、类...

    Java学习之IO总结及mina和netty

    这篇博客“Java学习之IO总结及mina和netty”探讨了Java IO的基础知识,并深入到两个高级网络通信框架——Mina和Netty。Mina和Netty都是基于NIO(非阻塞IO)的高性能网络应用框架,它们简化了复杂网络编程的实现。 *...

    Java JDK 6学习笔记——ppt

    Java JDK 6学习笔记——PPT简体版是针对初学者和有经验的开发者们的一份宝贵资源,它深入浅出地介绍了Java编程语言的核心概念和技术...无论你是初学者还是希望巩固Java基础的开发者,这都是一份非常有价值的参考资料。

    java基础知识详解 Java入门必看 Java面试题集锦

    这份资料集主要涵盖Java基础知识、初学者入门指南以及常见的Java面试题,旨在帮助学习者系统地理解和掌握Java的核心概念。 首先,Java基础知识是理解任何Java程序的基础。这包括: 1. **Java环境搭建**:学习如何...

    Java JDK 6学习笔记——ppt简体版附课本代码

    总的来说,“Java JDK 6学习笔记——ppt简体版”全面介绍了Java编程语言的基础知识和JDK 6的关键特性,结合配套的代码示例,是系统学习和掌握Java开发的宝贵资料。无论你是Java初学者还是寻求提升的老手,都能从中...

    Java实战开发(李兴华)——代码部分

    在本资源中,“Java实战开发(李兴华)——代码部分”主要涵盖了Java编程语言的实际应用和项目开发。李兴华是一位知名的Java教育专家,他的教程通常深入浅出,注重实践,适合初学者和有一定经验的开发者提升技能。...

    Java JDK 6学习笔记——ppt简体版.rar

    此外,`java.txt`可能包含了关于Java语言基础和进阶概念的文本资料,涵盖变量、数据类型、控制结构、类与对象、继承、多态等核心概念。这些内容对于理解和掌握Java编程至关重要。 通过"Java JDK 6学习笔记——ppt...

    Java基础知识课堂代码core java知识全集

    除了以上提到的IO、Swing和线程,Java基础知识还包括类与对象、封装、继承、多态、异常处理、集合框架(如ArrayList、LinkedList、HashMap等)、反射、注解、网络编程等。这些构成了Java程序员必须掌握的核心技能。...

    java初学者适用——java实例大全

    总之,"Java初学者适用——Java实例大全"是一个全面的教程,它不仅包含了Java语言的基础,还深入到高级主题,让初学者通过实践来深化理解,逐步成长为熟练的Java程序员。这份大全的每个实例都是一个学习的机会,通过...

    java课件(常用类)

    "java课件(常用类)"这个主题涵盖了Java基础知识中的重要部分——常用类。这些类是Java标准库的一部分,为开发者提供了丰富的功能,可以极大地提高开发效率。 在Java中,"常用类"通常指的是那些位于Java标准库...

    JAVA入门到精通——30天学通JAVA

    下面,我们将深入探讨Java编程的基础知识,涵盖关键概念、语法以及实用技巧。 首先,Java的基础部分包括编程环境的搭建。你需要下载并安装Java Development Kit (JDK),设置好环境变量,以便于编译和运行Java程序。...

    Java语言入门知识——为初学者提供入门基础

    Java语言是一种广泛应用于...通过学习以上基础知识,你将具备初步的Java编程能力。继续深入学习,你将能够开发更复杂的应用程序,如Web应用、数据库应用等。Java的世界广阔而精彩,欢迎踏入这个充满挑战和机遇的领域。

Global site tag (gtag.js) - Google Analytics