JavaIO之基础知识总结

一.流

1.0 概述

输入流
可以从其中读入一个字节序列的对象称作输入流

输出流
可以向其中写入一个字节序列的对象称作输出流

常见的流
文件、网络连接、内存块

1.1读写字节

InputStream和OutputStream是字节流体系的顶层抽象类。

基本方法
abstract int read()是InputStream类的一个抽象方法。这个方法将读入一个字节并返回读入的字节,在遇到输入源结尾时返回-1。

abstract void write(int b)是OutputStream类定义的一个抽象方法,它可以向某个输出位置写出一个字节。

具体的字节输入流,如FileInputStream类、System.in对象等都有自定义的read的方法,用于实现不同的功能。FileInputStream实现从文件中读取的read()方法,System.in的读取位置却是键盘。

同理,具体的字节输出流,也实现了各自的write方法。如FileOutputStream、System.out等。

扩展方法
基于read(),write()方法,顶层的InputStream和OutputStream还实现了int read(byte[] b)、int read(byte[] b,int off,int len)等方法和write(byte[] b)、write(byte[] b,int off,int len)等方法。

阻塞执行
read和write方法在执行时都将阻塞,直到字节确实被读入或写出。这意味着如果流不能被立即访问,那么当前线程将被阻塞。
读入输入流,InputStream.available()返回当前可读入的字节数量,即不需阻塞就能读取的字节数。

关闭流
不管是输入流还是输出流,使用完毕后都要调用close方法来关闭流。关闭流会释放掉十分有限的操作系统资源。关闭输出流的同时还会冲刷该输出流的缓冲区。当然也可以用flush方法来人为冲刷缓冲区。特别地,如果不关闭输出流,最后写出的字节可能会得不到传递。

1.2 完整的流家族

流整体上可分为字节流字符流两类。

字节流

顶层抽象类是InputStream和OutputStream,定义了基本read\write\close\flush\available等方法。

具体的实现类包括:

  • 可以从文件系统访问文件并进行读写文件的FileInputStream和FileOutputStream
  • 可以以二进制格式读写所有的基本Java类型的DataInputStream和DataOutputStream
  • 可以处理Zip文件的ZipInputStream和ZipOutputStream

字符流

字符流用于处理Unicode文本。
顶层抽象类是Reader和Writer。它俩的基本方法和字节流中的类似为:
abstract int read()
abstract void write(int c)
read方法将返回一个Unicode码元(一个在0-65535之间的整数),或者在碰到文件结尾时返回-1。write方法在被调用时,需要传递一个Unicode码元

流体系实现的接口

  • Closeable接口:定义了close()方法,四个顶层类都实现了它。
  • Flushable接口:定义了flush()方法,两个输出流OutpuStream和Writer实现了它
  • Readable接口:定义了int read(CharBuffer cb)方法,只有字符输入流Reader实现它。CharBuffer类拥有按顺序和随机地访问和读写的方法,它表示一个内存中的缓存区或者一个内存映像的文件??
  • Appendable接口:定义了Appendable append(char c)和Appendable append(CharSequence)两个方法,只有Writer实现了。可以方便地将String\StringBuffer\StringBuilder\CharBuffer等输出的流。

针对Appendable接口中提到的的CharSequence接口:String\StringBuffer\StringBuilder\CharBuffer等类都实现这个接口

1.3组合流过滤器

不同的流具有不同的能力,如FileInputStream可以从文件中读取数据,DataInputStream可以从流中读入数值类型。
可以组合不同的流,得到能力增强的流。如:要想从文件中读取二进制格式的数值类型,可以这么做:

DataInputStream din = new DataInputStream(new FileInputStream("data.dat"));
double s = din.readDouble();

再例如,流在默认情况下是不被缓冲区缓存的,也就是说对read的调用都会请求操作系统分发一个字节。相比之下,请求一个数据块并将其置于缓冲区中会显得更高效:

DataInputStream din = new DataInputStream(new BufferedInputStream(new FileInputStream("data.dat")));

PushbackInputStream介绍
可以回推字节。

PushbackInputStream pbin = new PushbackInputStream(new FileInputStream("data.dat"));
int b = pbin.read();
pbin.unread(b);

FileInputStream的构造方法

FileInputStream(String name);
FileInputStream(File file);

FileOutputStream的构造方法
如果append为true,则不会删除源文件,而是在源文件的末尾进行添加。

FileOutputStream(String name);
FileOutputStream(String name,boolean append);
FileOutputStream(File name);
FileOutputStream(File name,boolean append);

BufferedInputStream的构造方法

BufferedInputStream(InputStream in);

二.文本输入与输出

在保持数据时可以选择二进制格式或文本格式。例如整数1234存储为二进制数时,它写为字节00 00 04 D2(16进制法)。而存储成文本格式时,它被存成了字符串“1234”。尽管二进制格式的I/O高速且高效,但是不宜阅读。

先来介绍两个可以将“Unicode字符流”和“字节流”相互转换的类:
OutputStreamWriter类将使用选定的字符编码方式,把Unicode字符流转换为字节流;
InputStreamReader类将包含字节(用某种字符编码方法表示的字符)的输入流转换为可以产生Unicode码元的读入器。

InputStreamReader in = new InputStreamReader(new FileInputStream("data.dat"),"GBK");

文本格式的存储,本质上是将Unicode字符以某种编码方式编码为字节,再存储到文件中。读出时在以相同的编码方法解码后显示。

OutputStreamWriter和InputStreamReader就在其中起了关键的作用。

2.1如何写出到文本

可以使用PrintWriter类,它拥有以文本格式打印字符串和数字的方法。

构造方法
PrintWriter out = new PrintWriter("data.dat");
PrintWriter out = new PrintWriter(new FileWriter("data.dat"));
以上两个方法等价。

这里再多研究一下这两个方法:
查看FileWriter的构造方法:

    public FileWriter(String fileName) throws IOException {
        super(new FileOutputStream(fileName));
    }

首先用FileOutputStream获取文件的字节流,再传给父类。那再看看父类是谁?原来是OutputStreamWriter。这就呼应了前面说的OutputStreamWriter的作用:将使用选定的字符编码方式,把Unicode字符流转换为字节流

如何输出
out.print()、out.println()、out.printf()等方法
和System.out的输出类似

文本输出总结:
使用PrintWriter类。
如果是输出到文本,可以使用FileWriter获得一个字符输出流。而FileWriter是OutputStreamWriter的子类。本质上还是借用了OutputStreamWriter的字符流转字节流的能力。
另外,还可以指定PrintWriter的编码方式:

public PrintWriter(String fileName, String csn);//csn为字符集

2.2 如何读入文本输入

使用BufferedReader或Scanner。注意Scanner位于java.util包下。
首先借助于InputStreamReader获取字符流,再处理。InputStreamReader可以指定编码方式,从文件或其它字节流中得到字符流,再交给BufferedReader或Scanner进行读取处理。

1.BufferedReader

//FileInputStream从文件中读取字节流,InputStreamReader将字节流转换为字符流,BufferedReader带缓存的字符流
BufferedReader in = new BufferedReader(new InputStreamReader(new FileInputStream("data.txt"),"UTF-8"));

String line;
while((line = in.readLine()) != null) {
  //do something 
}

2.Scanner

    Scanner sc = new Scanner(new InputStreamReader(
            new FileInputStream("D://test_gb2312.txt"),
            "gb2312")); //这里可以指定字符集,用指定的这个字符集FileInputStream将字节码转化为Unicode字符
    int lineNum = 0;
    while (sc.hasNext()) {
        System.out.println("line " + (++lineNum) + " :");
        System.out.println(sc.nextLine());
    }
    sc.close();

2.3字符集

java.nio包下的Charset类统一了对字符集的转换。建立了两字节Unicode码元序列与使用本地字符编码方式的字节序列之间的映射。
可以使用静态方法Charset.forName("utf-8")来获取一个字符集
可以使用静态方法Charset.availableCharsets()来获取当前实现中可用的字符集合

三.读写二进制数据

概述
DataOutput接口定义了以下方法来以二进制形式写出字符串和8种java基本类型:
writeChars(String)写字符串 writeByte writeInt writeInt writeShort writeLong writeFloat writeDouble writeChar writeBoolean writeUTF

例如,writeInt总是将一个整数写出为4字节的二进制数量值,而不管它的大小。这样产生的结果不是可读的,但对于给定的类型的每个值其占用的空间都是一致的,而且将其读会也比解析文本要快。

类似地,DataInput接口中定义了读入8种基本类型的方法。

具体实现
DataInputStream和DataOutputStream实现上述两个接口。可以将它们和文件流组合来以二进制格式读写基本类型数据。

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

推荐阅读更多精彩内容

  • 概述 java.io 包几乎包含了所有操作输入、输出需要的类。所有这些流类代表了输入源和输出目标。java.io ...
    Steven1997阅读 9,194评论 1 25
  • 一、基础知识:1、JVM、JRE和JDK的区别:JVM(Java Virtual Machine):java虚拟机...
    杀小贼阅读 2,379评论 0 4
  • 五、IO流 1、IO流概述 (1)用来处理设备(硬盘,控制台,内存)间的数据。(2)java中对数据的操作都是通过...
    佘大将军阅读 508评论 0 0
  • 在经过一次没有准备的面试后,发现自己虽然写了两年的android代码,基础知识却忘的差不多了。这是程序员的大忌,没...
    猿来如痴阅读 2,841评论 3 10
  • 本文内容非原创,你可以点击此处查看内容来源声明 输入/输出流 在Java API中,可以从其中读出一个字节序列的对...
    _gitignore阅读 2,560评论 0 0