关于 I/O 的类可以分为四种:

  1. 关于字节的操作:InputStream 和 OutPutStream;
  2. 关于字符的操作:Writer 和 Reader;
  3. 关于磁盘的操作:File;
  4. 关于网络的操作:Socket( Socket 类不在 java.io 包中)。

在本篇博客中主要讲述前两种 I/O,即字符流与字节流,并会提及磁盘IO。首先我们来看一下字节流与字符流的实现关系,如下图

一、字节流

在字节流的类中,最顶层的是 Inputstream 抽象类和 OutputStream 抽象类,两者定义了一些关于字节数据读写的基本操作。他们的实现类有

  • ByteArrayInputStream/ByteArrayOutputStream // 都对“字节数组”进行操作(写入/读出),两者可单独使用
  • FileInputStream/FileOutputStream // 都对“文件”进行操作(写入/读出),两者可单独使用
  • ObjectInputStream/ObjectOutputStream // 用于“对象”与“流”之间的转换,两者可单独使用
  • PipedInputStream/PipedOutputStream // 建立一个链接(像管道),用于双方传输流数据,两者要一起使用
  • FilterInputStream/FilterOutputStream // 装饰类,可以为其他类增加功能,应用了装饰者模式
    • BufferedInputStream/BufferedOutputStream // 可以为被装饰的类增加缓冲功能
    • DataInputStream/DataOutputStream // 可以为被装饰的类增加写读指定数据类型的功能
    • LineNumberInputStream // 可以为被装饰的类增加获取行数的功能,现已被 LineNumberReader 替代
    • PushbackInputStream // 可以将当前读取的数据推回到缓冲区
    • PrintStream
  • SequenceInputStream
  • StringbufferInputStream // 将字符转换为字节,推荐使用 StringReader

1 InputStream类与OutputStream类

InputStream 抽象类中的方法如下:

int read() //读取一个字节
int read(byte b[]) //最多读取 b.length 长度的字节,并储存到数组 b 中,返回实际读取长度,利用 read() 实现
int read(byte b[], int off, int len) //同上,最多读取 len 个字节,从 off 开始保存
int available() //返回可以读取的字节数目,不受阻塞
long skip(long n) //跳过n个字节
void close() //关闭输入流,并释放与此流有关的所有资源,未实现,方法体为空
synchronized void mark(int readlimit) //标记当前位置,以便可以使用 reset() 方法复位到该标记的位置
synchronized void reset() //将当前位置复位到上次使用 mark() 方法标记的位置,未实现,调用抛出IO异常
boolean markSupported() //判断次输入流是否支持 mark() 和 reset() 方法

OutputStream 抽象类中的方法如下:

void write(int b) // 抽象的写入方法
void write(byte b[]) // 写入一个字节数组,利用 write(int b)实现
void write(byte b[], int off, int len) // 同上
void flush()
void close()

1.1 ByteArrayInputStream类与ByteArrayOutputStream类

ByteArrayInputStream 与 ByteArrayOutputStream 都对“字节数组”的操作,将数据写入数组、或从数组读出数据。两者并无直接关系,都可以单独使用。

1) ByteArrayInputStream

创建 ByteArrayInputStream 对象时,需要一个“字节数组”,并且会存入其内部的缓冲区 (byte数组) 以待读取,我们可以通过 read() 方法来获取。

在 ByteArrayInputStream 内部还有几个属性:

  • int pos 当前读取到的位置
  • int mark 记录复位的位置
  • int count 带读取字符总长度

本类中的方法相对其父类的方法也没有增加,使用也较为简单,就不再描述。另外,我们可以利用 ByteArrayInputStream 来读取 ByteArrayOutputStream 写入的数据,如下图

  

2) ByteArrayOutputStream

创建 ByteArrayOutputStream 时,可以指定其内部缓冲区 (byte数组) 的大小 (默认32字节),我们可以通过 wirte() 方法向缓冲区写入数据,如果缓冲区长度不够时,会自动扩大至少一倍。

在 ByteArrayOutputStream 内部还有几个属性:

  • int count 用来记录缓冲区的大小
  • static final int MAX_ARRAY_SIZE 缓冲区的最大值

如果需要使用这些数据时我们可以通过 toString() 、toByteArray() 从缓冲区中获取,也可以通过 writeTo() 将缓冲区数据其写入其它 OutputStream 类中,如下图

ByteArrayOutputStream 的部分方法如下:

synchronized void write(int b) // 向缓冲区写入数据
synchronized void write(byte b[], int off, int len)
synchronized void writeTo(OutputStream out) //将缓冲区数据写入到其他输出流中
synchronized void reset() //重置缓冲区
synchronized byte toByteArray()[]synchronized String toString()
synchronized String toString(String charsetName)
synchronized String toString(int hibyte) // @Deprecated 已弃用

1.2 FileInputStream类与FileOutputStream类

FileInputStream 与 FileOutputStream 都对“文件”的操作,将数据写入文件、或从文件读出数据。两者并无直接关系,都可以单独使用。与 ByteArrayInputStream 和 ByteArrayOutputStream 类似,但不同的是一个操作文件,一个操作字符数组。

1) FileInputStream

创建 FileInputStream 对象时,我们可以传入文件的路径、对象或者描述符对象。此时会检查文件是否为空、是否有效、是否可读 (使用 System.getSecurityManager() 获取安全管理器来检查),然后调用本地方法打开文件。

在 FileInputStream 中还有几个属性用来描述文件信息:

  • FileDescriptor 文件的描述符
  • String path 存储了文件的路径
  • FileChannel

当我们从文件中读取数据时,FileInputStream 会再去调用本地方法去实现此操作。FileInputStream 部分方法如下:

int read() // 读取1个字节,如果已经读完返回-1,调用native read0()来实现,不会阻塞
int read(byte b[]) // 读取到字节数组中,调用native readBytes()来实现,不会阻塞
int read(byte b[],int off, int len)
native long skip(long n) // 跳过n个字节
native int available()
FileDescriptor getFD() // 返回表示到文件系统中实际文件的连接的 FileDescriptor 对象。
FileChannel getChannel() // 返回与此文件输入流有关的唯一FileChannel 对象
void close() //关闭流,用closeLock变量作为锁在synchronized代码块中改变closed属性,再调用本地方法关闭
void finalize() // 确保在不再引用文件输入流时调用其close()方法

2) FileOutputStream

创建 FileOutputStream 对象时,我们可以传入文件的路径、对象或者描述符对象,并且可以再传入一个 boolean 值,判断是否要以追加的方式写入。同样,要判断文件是否为空、是否有效、是否可写,然后调用本地方法打开文件。

在 FileOutputStream 中的部分属性如下:

  • 描述符 FileDescriptor
  • 路径
  • FileChannel

当我们向文件中写入数据时,FileOutputStream 会再去调用本地方法去实现此操作。FileOutputStream 部分方法如下:

void write(int b) // 向文件中写入一个字节,调用native write()来实现
void write(byte b[]) // 向文件中写入一个字节数组,调用native writeBytes()来实现
void write(byte b[], int off, int len) // 同上
void close() //关闭流,用closeLock变量作为锁在synchronized代码块中改变closed属性,再调用本地方法关闭
FileDescriptor getFD()
FileChannel getChannel()
void finalize() //

1.3 ObjectInputStream类和ObjectOutputStream类

ObjectInputStream 和 ObjectOutputStream 用于“对象”与“流”之间的转换,从流中读出对象或将对象写入流中(这里的流可能是来自文件,也可能是来自网络)。另外,该类所读写的对象必须实现Serializable接口.

1) ObjectInputStream

创建 ObjectInputStream 对象时,需要传入一个流(如 FileInputStream),如下图

在 ObjectInputStrean 中的内部类

  • private static class Caches
  • public static abstract class GetField
  • private class GetFieldImpl extends GetField
  • private static class ValidationList
  • private static class PeekInputStream extends InputStream
  • private class BlockDataInputStream extends InputStream implements DataInput
  • private static class HandleTable

ObjectInputStream 中的部分方法如下

Object readObject()
Object readUnshared()
void defaultReadObject()
ObjectInputStream.GetField readFields()
void registerValidation(ObjectInputValidation obj, int prio)int read()
int read(byte[] buf, int off,int len)
int available()
void close()
boolean readBoolean()
byte readByte()
int readUnsignedByte()
char readChar()
short readShort()
int readUnsignedShort()
int readInt()
long readLong()
float readFloat()
double readDouble()
void readFully(byte[] buf)
void readFully(byte[] buf, int off,int len)
int skipBytes(int len)
public String readUTF()

2) ObjectOutputStream

创建 ObjectOutputStream 对象时,传入一个流(如 FileOutputStream),如下图

在 ObjectOutputStream 中的内部类

  • private static class Caches
  • public static abstract class PutField
  • private class PutFieldImpl extends PutField
  • private static class BlockDataOutputStream extends OutputStream implements DataOutput
  • private static class HandleTable
  • private static class ReplaceTable
  • private static class DebugTraceInfoStack

1.4 PipedInputStream类与PipedOutputStream类

PipedInputStream 和 PipedOutputStream 的实现原理类似于“生产者-消费者”原理,PipedOutputStream 是生产者,PipedInputStream 是消费者,在 PipedInputStream 中有一个 buffer 字节数组作为缓冲区,默认大小为1024,存放“生产者”生产出来的东西。还有两个变量 in 和 out,in 是用来记录“生产者”生产了多少,out 是用来记录“消费者”消费了多少,in 为-1表示消费完了,in==out 表示生产满了。当消费者没东西可消费的时候,也就是当 in 为-1的时候,消费者会一直等待,直到有东西可消费。

PipedInputStream 和 PipedOutputStream 数据传输的过程:

  1. 首先两者要建立一个完整管道,实现方式是“输入流”调用“输出流”的 connect() 方法并将自己传入。在这个方法中“输出流”拿到了“输入流”的对象,然后将其传入 sink 变量,并设置“输入流”的 connected 属性为 true。
  2. 建立好完整管道后就可以传输数据了,因为刚建立好的管道里并没有缓存数据,所以首先向管道中写入数据。我们使用“输出流”的 write() 方法来写入数据,在此方法中会调用“输入流”的 receive() 方法并将数据传入。如果此时“输入流”线程在等待状态,可以调用“输出流”的 flush() 方法,这个方法中会调用“输入流”的 notifyAll() 方法将其唤醒。当数据通过“输入流”的 receive() 方法进入“输入流”后,将其放入缓冲区中以待读取。
  3. 当数据写入到“输入流”的缓冲区后,“输入流”就可以使用 read() 方法读取数据了。当“输入流”与“输出流”使用完毕后,应调用双方的 close() 方法断开链接。

1) PipedInputStream

创建 PipedInputStream 对象时,可以根据 PipedOutputStream 对象来建立一个完整的管道链接,并设置缓冲区大小(默认1024)。如果创建对象时不建立连接,那么也要在使用前将链接建立好。

我们来看一下部分属性

  • boolean closedByWriter 与 volatile boolean closedByReader 用来记录管道是否关闭
  • connected 用来记录当前管道是否链接
  • Thread readSide 与 Thread writeSide;
  • byte buffer[] 缓冲区
  • int in 初始为-1,表示从输出管道读出下一个字节后在缓冲区储存的位置,in==-1表示缓冲区为空,in==out表示缓冲区已满
  • int out 初始为0,表示从缓冲区读取下一个字节的位置

PipedInputStream相关方法:

private void initPipe(int pipeSize) //初始化缓冲区
public void connect(PipedOutputStream src) //与输出管道建立连接
protected synchronized void receive(int b) //接收一个字节的数据,没有数据输入则阻塞
synchronized void receive(byte b[], int off, int len) //接收大小为len的数据到数组,不足则阻塞
private void checkStateForReceive() //判断是否建立连接,读写是否已关闭
private void awaitSpace()synchronized void receivedLast() //通知所有等待的线程,最后一个字符已经接收
public synchronized int read() //从缓冲区读取字节,返回读取的字节,如果缓冲区没有数据则会阻塞
public synchronized int read(byte b[], int off, int len) //同上
public synchronized int available()
public void close()

2) PipedOutputStream

创建 PipedOutputStream 对象时,同样可以传入一个 PipedInputStream 对象,然后来建立一个完整的管道连接,如果不建立连接,那要在使用前将链接建立好。PipedOutputStream 没有缓存区,写入的数据会进入到 PipedInputStream 的缓存区中。

PipedOutputStream 只有一个属性:PipedInputStream sink 用来存放所链接的输入流对象。

PipedOutputStream相关方法:

public synchronized void connect(PipedInputStream snk) // 与输入流建立连接,需要先获取到对象锁
public void write(int b) // 调用连接的输入流的receive方法,向其写入一个字节
public void write(byte b[], int off, int len) // 同上,写入一个字节数组
public synchronized void flush() // 唤醒所连接的输入流
public void close()

关于 PipedInputStream 和 PipedOutputStream 的几点要注意:

  • 两者建立连接时,不能既调用 PipedInputStream 的 connect() 方法又调用 PipedOutputStream 的 connect() 方法,否则会抛出IO异常。且 PipedInputStream 实现 connect() 方法是去调用 PipedOutputStream 的 connect() 方法。
  • 在一个线程里使用 PipedInputStream 和 PipedOutputStream 容易造成卡死(“死锁”)。例如,当我们调用 PipedOutputStream 的 write() 方法写入1024字节数据时,该方法通过调用 PipedInputStream 的 receive() 方法将字符数组中的数据写入到缓冲区,我们在读取128字节数据后,继续再向缓冲区写入1024字节数据,会发现程序卡在写入数据的这个过程中,造成死锁。原因是我们第二次向缓冲区写入数据时,缓冲区满了,这时候 PipedInputStream 的 awaitSpace() 方法会去执行 notifyAll() 试图通知 read() 方法去读取数据,在缓冲区的数据被读取之前,write() 方法会一直阻塞下去。又因为 PipedInputStream 和 PipedOutputStream 在一个线程里,write() 方法的阻塞导致 read() 方法不能去缓冲区读取数据,从而形成“死锁”。

      产生“死锁”的代码:                   输出结果: 

          

1.5 FilterInputStream类与FilterOutputStream类及它们的子类

FilterInputStream 与 FilterOutputStream 及子类是使用了装饰者设计模式。为什么要使用装饰者设计模式?如果我们觉得某几个类都缺少一些功能的时候,根据开闭原则,我们不去修改代码,那我们可以利用继承对功能进行扩展,但是如果几个类都需要扩展相同的功能,那么就必须分别对这个几个类扩展。如果使用装饰者设计模式,我们就可以解决这个问题。举个例子,当我们使用 ByteArrayInputStream 的时候只能读取到以 byte 为单位的数据,我们可以按需要把读取出来的字节进行转码,转成我们需要的数据,那么能不能直接读取int、long等数据呢?答案是可以的,我们利用 DataInputStream 类对其进行装饰即可拥有 readInt()、readLong() 这个方法。

1) FilterInputStream

创建 FilterInputStream 时需要传入被修饰的输入流对象,并使用 volatile InputStream in 记录该对象。FilterInputStream 是所有输入流修饰类的父类,但其本身并没有修饰作用,而是直接使用被修饰类的方法,如下是类中的方法

public int read()
public int read(byte b[])
public int read(byte b[], int off, int len)
public long skip(long n)
public int available()
public void close()
public synchronized void mark(int readlimit)
public synchronized void reset()
public boolean markSupported()

2) FilterOutputStream

创建 FilterOutputStream 时需要传入被修饰的输出流对象,并使用 OutputStream out 记录该对象。FilterOutputStream 是所有输出流修饰类的父类,其本身也没有修饰作用,如下是类中的方法

public void write(int b)
public void write(byte b[])
public void write(byte b[], int off, int len)
public void flush()
public void close()

1.5.1 DataInputStream类与DataOutputStream类

DataInputStream 与 DataOutputStream 是装饰类,可以对继承了 InputStream 的类进行装饰,使其用于读取或写入指定类型内容的数据。

1) DataInputStream

创建时需要传入被包装的输入流对象,相关方法如下

public final int read(byte b[]) // 调用被装饰对象的read()方法
public final int read(byte b[], int off, int len) // 同上
public final void readFully(byte b[]) // 读取数据到数组,直到存满,当流中数据不足时,readFully()会阻塞等待
public final void readFully(byte b[], int off, int len) // 同上
public final int skipBytes(int n) // 跳过n个字节
public final boolean readBoolean()
public final byte readByte()
public final int readUnsignedByte()
public final short readShort() // 连续调用read()两次,然后将结果拼接后强转short类型,(short)((ch1<<8)+(ch2<<0))
public final int readUnsignedShort() // 同上,但不需要进行强转
public final char readChar()
public final int readInt()
public final long readLong() // 利用readFully()读取8个字节到readBuffer数组,然后拼接转换成long类型
public final float readFloat() // 使用Float.intBitsToFloat(readInt())
public final double readDouble() // 使用Double.longBitsToDouble(readLong())
public final String readLine() // 弃用的
public final String readUTF()
public final static String readUTF(DataInput in) // 待研究

2) DataOutputStream

DataOutputStream 相关方法如下

private void incCount(int value) // 记录写入到流的字节数
public synchronized void write(int b)
public synchronized void write(byte b[], int off, int len)
public void flush() // 让缓存中的数据传入到输入流中
public final void writeBoolean(boolean v)
public final void writeByte(int v)
public final void writeShort(int v)
public final void writeChar(int v)
public final void writeInt(int v)
public final void writeLong(long v)
public final void writeFloat(float v) // writeInt(Float.floatToIntBits(v))
public final void writeDouble(double v) // writeLong(Double.doubleToLongBits(v));
public final void writeBytes(String s)
public final void writeChars(String s)
public final void writeUTF(String str)
static int writeUTF(String str, DataOutput out)
public final int size()

1.5.2 BufferedInputStream类与BufferedOutputStream类

当我们在使用 FileInputStream 和 FileOutputStream 时,要用一个 byte 数组用来接收或写入数据,硬盘存取的速度远低于内存中的数据存取速度,为了减少对硬盘的存取,通常从文件中一次读入一定长度的数据,而写入时也是一次写入一定长度的数据,这可以增加文件存取的效率。

1) BufferedInputStream

BufferedInputStream 有一个字节数组 buf 作为缓冲区,默认为8192字节。当使用 read() 方法读取数据时,实际上是先读取 buf 中的数据,当 buf 中的数据不足时,BufferedInputStream 会再从被装饰的 InputStream 对象的 read() 方法中读取数据填满 buf ,然后再从 buf 中读取所需大小的数据。如果一次读取的数据大小超过 buf 缓冲区的大小,则放弃缓冲直接调用被装饰对象的 read() 方法。

另外,BufferedInputStream 利用原子更新字段类 (AtomicReferenceFieldUpdater) 对缓冲区 volatile byte buf[] 进行包装。当我们对缓冲区扩容时,得到一个新的缓冲区数组后替换旧的缓冲区数组,此时就要利用原子更新字段类更新。

/*
AtomicReferenceFieldUpdater 位于Atomic包中,是一个基于反射的工具类,它能对指定类的指定的volatile引用字段进行原子更新(注意这个字段不能是private的)。通过静态方法newUpdater就能创建它的实例,三个参数分别是:包含该字段的对象的类、将被更新的对象的类、将被更新的字段的名称。
*/
private static final
AtomicReferenceFieldUpdater<BufferedInputStream, byte[]> bufUpdater =
AtomicReferenceFieldUpdater.newUpdater
(BufferedInputStream.class, byte[].class, "buf");
protected int count; // 缓冲区内容大小
protected int pos; // 缓冲区当前位置
protected int markpos = -1;
protected int marklimit;
// 该类中的方法:
private InputStream getInIfOpen() // 检查流是否存在
private byte[] getBufIfOpen() // 检查缓冲区是否存在
private void fill() //
public synchronized int read()
private int read1(byte[] b, int off, int len)
public synchronized int read(byte b[], int off, int len)
public synchronized long skip(long n)
public synchronized int available()
public synchronized void mark(int readlimit)
public synchronized void reset()
public boolean markSupported()
public void close()

2) BufferedOutputStream

BufferedOutputStream 也有一个字节数组 buf 作为缓冲区,默认同样为8192字节,并用一个变量 count 记录缓冲区内数据大小。当使用 write() 方法写入数据时,实际上会先将数据写至 buf 中,当 buf 已满时,BufferedOutputStream 才会利用被装饰的 OutputStream 对象的 write() 方法写入数据。如果一次写入的数据大小超过 buf 缓冲区的大小,则放弃缓冲直接调用被装饰对象的 write() 方法。

BufferedOutputStream 的相关方法如下

private void flushBuffer() // 将缓冲区的数据写入流,调用了被装饰对象的write方法
public synchronized void write(int b) // 将数据写入缓冲,如果缓冲满了,调用flushBuffer()方法
public synchronized void write(byte b[], int off, int len) // 同上
public synchronized void flush()

1.5.3 PushbackInputStream类

对输入流进行装饰,可以将当前读取的字节数据推回到缓存区,一般用不到的。

// 该类中的属性:
protected byte[] buf;
protected int pos;
// 该类中的方法:
private void ensureOpen()
public int read()
public int read(byte[] b, int off, int len)
public void unread(int b)
public void unread(byte[] b, int off, int len)
public void unread(byte[] b)
public int available()
public long skip(long n)
public boolean markSupported() // 不支持mark功能
public synchronized void mark(int readlimit)
public synchronized void reset()
public synchronized void close()

1.5.4 LineNumberInputStream类

对输入流进行装饰,可以获取输入流的行数或设置行数,已过时。已经被LineNumberReader替代。

1.5.5 PrintStream类

对输出流进行装饰,

// 构造方法:
private PrintStream(boolean autoFlush, OutputStream out) // 传入一个需装饰的对象
private PrintStream(boolean autoFlush, OutputStream out, Charset charset)
private PrintStream(boolean autoFlush, Charset charset, OutputStream out)
public PrintStream(OutputStream out)
public PrintStream(OutputStream out, boolean autoFlush)
public PrintStream(OutputStream out, boolean autoFlush, String encoding)
public PrintStream(String fileName)
public PrintStream(String fileName, String csn)
public PrintStream(File file)
public PrintStream(File file, String csn)
// 该类中的属性:
private final boolean autoFlush;
private boolean trouble = false;
private Formatter formatter;
private BufferedWriter textOut;
private OutputStreamWriter charOut;
private boolean closing = false;
// 该类中的方法:
private static <T> T requireNonNull(T obj, String message)
private static Charset toCharset(String csn)
private void ensureOpen()
public void flush()
public void close()
public boolean checkError()
protected void setError()
protected void clearError()
public void write(int b)
public void write(byte buf[], int off, int len)
private void write(char buf[])
private void write(String s)
private void newLine()
public void print(boolean b)
public void print(char c)
public void print(int i)
public void print(long l)
public void print(float f)
public void print(double d)
public void print(char s[])
public void print(String s)
public void print(Object obj)
public void println()
public void println(boolean x)
public void println(char x)
public void println(int x)
public void println(long x)
public void println(float x)
public void println(double x)
public void println(char x[])
public void println(String x)
public void println(Object x)
public PrintStream printf(String format, Object ... args)
public PrintStream printf(Locale l, String format, Object ... args)
public PrintStream format(String format, Object ... args)
public PrintStream format(Locale l, String format, Object ... args)
public PrintStream append(CharSequence csq)
public PrintStream append(CharSequence csq, int start, int end)
public PrintStream append(char c)

1.6 StringBufferInputStream类

已经不再适合将字符转化为字节,更优的选择是通过StringReader将字符转化为流。

1.7 SequenceInputStream类

// 构造方法:
public SequenceInputStream(Enumeration<? extends InputStream> e)
public SequenceInputStream(InputStream s1, InputStream s2)
// 该类中的属性:
Enumeration<? extends InputStream> e
InputStream in;
// 该类中的方法:
final void nextStream()
public int available()
public int read()
public int read(byte b[], int off, int len)
public void close()

二、Reader类与Writer类

在字符流的类中,最顶层的是 Reader 抽象类和 Writer 抽象类,两者定义了一些关于字符数据读写的基本操作。他们的实现类有

  • CharArrayReader/CharArrayWriter
  • StringReader/StringWriter
  • InputStreamReader/OutputStreamWriter
    • FileReader/FileWriter
  • PipedReader/PipedWriter
  • FilterReader/FilterWriter
    • PushbackReader
  • BufferedReader/BufferedWriter
    • LineNumberReader
  • PrintWriter

2、Reader类与Writer类及它们的子类

Reader抽象类中的方法如下:

// 构造方法
protected Reader()
protected Reader(Object lock)
// 该类中的属性
protected Object lock;
private static final int maxSkipBufferSize = 8192;
private char skipBuffer[] = null;
// 该类中的方法
public int read(java.nio.CharBuffer target)
public int read()
public int read(char cbuf[])
abstract public int read(char cbuf[], int off, int len)
public long skip(long n)
public boolean ready()
public boolean markSupported()
public void mark(int readAheadLimit)
public void reset()
abstract public void close()

Writer抽象类中的方法如下:

// 构造方法
protected Writer()
protected Writer(Object lock)
// 该类中的属性
private char[] writeBuffer;
private static final int WRITE_BUFFER_SIZE = 1024;
protected Object lock;
// 该类中的方法
public void write(int c)
public void write(char cbuf[])
abstract public void write(char cbuf[], int off, int len)
public void write(String str)
public void write(String str, int off, int len)
public Writer append(CharSequence csq)
public Writer append(CharSequence csq, int start, int end)
public Writer append(char c)
abstract public void flush()
abstract public void close()

2.1 CharArrayReader类与CharArrayWriter类

CharArrayReader 与 CharArrayWriter,并无直接关系,两者都可以单独使用。

CharArrayReader相关方法:

// 构造方法
public CharArrayReader(char buf[])
public CharArrayReader(char buf[], int offset, int length)
// 该类中的属性
protected char buf[]; // 缓冲区
protected int pos; // 缓冲区使用位置
protected int markedPos = 0;
protected int count;
// 该类中的方法
private void ensureOpen() // 如果流关闭,则抛出IO异常
public int read() // 读一个字符,返回,若流中数据已读完则返回-1
public int read(char b[], int off, int len)
public long skip(long n)
public boolean ready() // 判断是否已经可以从流中读数据
public boolean markSupported()
public void mark(int readAheadLimit)
public void reset()
public void close()

CharArrayWriter相关方法:

// 构造方法
public CharArrayWriter() // 缓冲区默认32
public CharArrayWriter(int initialSize)
// 该类中的属性
protected char buf[]; // 缓冲区
protected int count;
// 该类中的方法
public void write(int c)
public void write(char c[], int off, int len)
public void write(String str, int off, int len)
public void writeTo(Writer out) // 将缓冲区的内容写入到另一个字符流
public CharArrayWriter append(CharSequence csq) //
public CharArrayWriter append(CharSequence csq, int start, int end) //
public CharArrayWriter append(char c) //
public void reset()
public char toCharArray()[]
public int size()
public String toString()
public void flush()
public void close()

2.2 StringReader类与StringWriter类

与CharArrayReader 与 CharArrayWriter 相似

StringReader相关方法:

// 构造方法
public StringReader(String s)// 该类中的属性
private String str;
private int length;
private int next = 0;
private int mark = 0;
// 该类中的方法
private void ensureOpen()
public int read()
public int read(char cbuf[], int off, int len)
public long skip(long ns)
public boolean ready()
public boolean markSupported()
public void mark(int readAheadLimit)
public void reset()
public void close()

StringWriter相关方法(就是把StringBuffer封装了一下):

// 构造方法
public StringWriter()
public StringWriter(int initialSize)
// 该类中的属性
private StringBuffer buf;
// 该类中的方法
public void write(int c)
public void write(char cbuf[], int off, int len)
public void write(String str)
public void write(String str, int off, int len)
public StringWriter append(CharSequence csq)
public StringWriter append(CharSequence csq, int start, int end)
public StringWriter append(char c)
public String toString()
public StringBuffer getBuffer()
public void flush()
public void close()

2.3 PipedReader类与PipedWriter类

PipedReader 与 PipedWriter 的使用和 PipedInputStream 与 PipedOutputStream 十分相似,原理也相同。

PipedReader相关方法:

// 构造方法
public PipedReader(PipedWriter src)
public PipedReader(PipedWriter src, int pipeSize)
public PipedReader()
public PipedReader(int pipeSize)
// 该类中的属性
boolean closedByWriter = false;
boolean closedByReader = false;
boolean connected = false;
Thread readSide;
Thread writeSide;
private static final int DEFAULT_PIPE_SIZE = 1024;
char buffer[];
int in = -1;
int out = 0;
// 该类中的方法
private void initPipe(int pipeSize)
public void connect(PipedWriter src)
synchronized void receive(int c)
synchronized void receive(char c[], int off, int len)
synchronized void receivedLast()
public synchronized int read()
public synchronized int read(char cbuf[], int off, int len)
public synchronized boolean ready()
public void close()

PipedWriter相关方法:

// 构造方法
public PipedWriter(PipedReader snk)
public PipedWriter()
// 该类中的属性
private PipedReader sink;
private boolean closed = false;
// 该类中的方法
public synchronized void connect(PipedReader snk)
public void write(int c)
public void write(char cbuf[], int off, int len)
public synchronized void flush()
public void close()

2.4 InputStreamReader类与OutputStreamWriter类

InputStreamReader 与 OutputStreamWriter 使用的是适配器模式,作为字节流与字符流两个不兼容的类(接口)之间的桥梁。以 InputStreamReader 为例,InputStreamReader 继承了 Reader 抽象类,利用 InputStream 对象(这个对象是由构造方法传来的)的方法,对 Reader 抽象类的方法进行实现(实现的时候是利用 StreamDecoder 进行转码),从而使字节流转换成字符流。

InputStreamReader相关方法:

// 构造方法
public InputStreamReader(InputStream in)
public InputStreamReader(InputStream in, String charsetName)
public InputStreamReader(InputStream in, Charset cs)
public InputStreamReader(InputStream in, CharsetDecoder dec)
// 该类中的属性
private final StreamDecoder sd;
// 该类中的方法
public String getEncoding()
public int read()
public int read(char cbuf[], int offset, int length)
public boolean ready()
public void close()

OutputStreamWriter相关方法:

// 构造方法
public OutputStreamWriter(OutputStream out, String charsetName)
public OutputStreamWriter(OutputStream out)
public OutputStreamWriter(OutputStream out, Charset cs)
public OutputStreamWriter(OutputStream out, CharsetEncoder enc)
// 该类中的属性
private final StreamEncoder se;
// 该类中的方法
public String getEncoding()
void flushBuffer()
public void write(int c)
public void write(char cbuf[], int off, int len)
public void write(String str, int off, int len)
public void flush()
public void close()

2.4.1 FileReader类和FileWriter类

两者都是在构造方法中创建对应的 FileInputStream 或 FileOutputStream对象,然后将调用父构造方法将其传入。

public FileReader(String fileName)
public FileReader(File file)
public FileReader(FileDescriptor fd)
public FileWriter(String fileName)
public FileWriter(String fileName, boolean append)
public FileWriter(File file)
public FileWriter(File file, boolean append)
public FileWriter(FileDescriptor fd)

2.5 FilterReader类和FilterWriter类

FilterReader相关方法:

// 构造方法
protected FilterReader(Reader in)
// 该类中的属性
protected Reader in; // 需要装饰的对象
// 该类中的方法
public int read()
public int read(char cbuf[], int off, int len)
public long skip(long n)
public boolean ready()
public boolean markSupported()
public void mark(int readAheadLimit)
public void reset()
public void close()

FilterWriter相关方法:

// 构造方法
protected FilterWriter(Writer out)
// 该类中的属性
protected Writer out; // 需要装饰的对象
// 该类中的方法
public void write(int c)
public void write(char cbuf[], int off, int len)
public void write(String str, int off, int len)
public void flush()
public void close()

2.5.1 PushbackReader类

// 构造方法
public PushbackReader(Reader in, int size)
public PushbackReader(Reader in)
// 该类中的属性
private char[] buf;
private int pos;
// 该类中的方法
private void ensureOpen()
public int read()
public int read(char cbuf[], int off, int len)
public void unread(int c)
public void unread(char cbuf[], int off, int len)
public void unread(char cbuf[])
public boolean ready()
public void mark(int readAheadLimit)
public void reset()
public boolean markSupported()
public void close()
public long skip(long n)

[Java] I/O底层原理之一:字符流、字节流及其源码分析的更多相关文章

  1. Java IO(十二) 字符流 Writer 和 Reader

    Java IO(十二) 字符流 Reader和 Writer 一.介绍 涉及到文件(如果是纯文本文件形式)操作时,Java除了提供 FIle(文件和目录路径名的抽象表示形式) 和 FileDescri ...

  2. java 字符流 字节流

    java对文本文档进行操作(拷贝.显示)出现乱码一般来说,可以从两个方面入手. 1.文本文件本身的编码格式. 2.java代码中处理文本文件的编码格式. 这里要注意的一点是,我们可以看出copyFil ...

  3. Hadoop之HDFS原理及文件上传下载源码分析(下)

    上篇Hadoop之HDFS原理及文件上传下载源码分析(上)楼主主要介绍了hdfs原理及FileSystem的初始化源码解析, Client如何与NameNode建立RPC通信.本篇将继续介绍hdfs文 ...

  4. vue双向绑定的原理及实现双向绑定MVVM源码分析

    vue双向绑定的原理及实现双向绑定MVVM源码分析 双向数据绑定的原理是:可以将对象的属性绑定到UI,具体的说,我们有一个对象,该对象有一个name属性,当我们给这个对象name属性赋新值的时候,新值 ...

  5. java IO之 字符流 (字符流 = 字节流 + 编码表) 装饰器模式

    字符流 计算机并不区分二进制文件与文本文件.所有的文件都是以二进制形式来存储的,因此, 从本质上说,所有的文件都是二进制文件.所以字符流是建立在字节流之上的,它能够提供字符 层次的编码和解码.列如,在 ...

  6. 深入理解JAVA I/O系列三:字符流详解

    字符流为何存在 既然字节流提供了能够处理任何类型的输入/输出操作的功能,那为什么还要存在字符流呢?容我慢慢道来,字节流不能直接操作Unicode字符,因为一个字符有两个字节,字节流一次只能操作一个字节 ...

  7. Java当中的I/O的字符流

    字符流读写文件时,以字符为基础 I/O当中字符流的核心类 Reader类和Writer类是所有字符流类的父类,同样也是抽象类.FileReader和FileWriter分别是它们的子类. 核心类的核心 ...

  8. 【Java】 剑指offer(50-2) 字符流中第一个只出现一次的字符

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 请实现一个函数用来找出字符流中第一个只出现一次的字符.例如,当从字 ...

  9. Java学习笔记29(IO字符流,转换流)

    字符流:只能操作文本文件,与字节流的区别是,字节流是按照字节来读取文件,而字符流是按照字符来读取,因此字符流的局限性为文本文件 字符输出流:Write类,使用时通过子类   每一次写入都要刷新 pac ...

随机推荐

  1. Splinter学习——不仅仅是自动化测试哦

    前两天,想抢购一个小米MIX,结果,一开始抢就没有了.于是想,作为程序猿,总得有点特殊手段吧,比如说一个小脚本.最近在学习python,百度了一下,发现了Splinter这个强大的东东!用了不到两小时 ...

  2. 使用SignalR实现即时通讯功能

    教程简介 SignalR的好处是可以让多个客户端之间进行互动,比如这篇教程就展示了当你在页面上拖动矩形方块的同时,其它打开这个页面的用户也将会看到你拖动的轨迹以及最终的结果,当然他们也可以通过拖动该方 ...

  3. 微软职位内部推荐-Software Engineer II-SDP

    微软近期Open的职位: Position: SDE II The R&D of Shared Data Platform at Application and Services Group ...

  4. IBInspectable的使用

    IBInspectable的使用 创建一个分类 定义属性 把此分类的头文件引入到pch里面  此时查看属性面板  就有了可以供我们勾选的选项 在.m中实现一下set和get方法

  5. [POJ] 2226 Muddy Fields(二分图最小点覆盖)

    题目地址:http://poj.org/problem?id=2226 二分图的题目关键在于建图.因为“*”的地方只有两种木板覆盖方式:水平或竖直,所以运用这种方式进行二分.首先按行排列,算出每个&q ...

  6. windows系统各版本 各种数据结构

    极爽啊http://msdn.moonsols.com/

  7. iOS Storyboard约束详解

    链接:http://www.jianshu.com/p/b88c65ffc3eb 约束,就是指--此处略去1万字--都懂的,就不说了.直接进入实战环节. 本文的控件约束都是围绕着UITableView ...

  8. Android基础Activity篇之什么是Activity?

    Activity是Android的四大组件之一,也是平时我们用到最多的一个组件,可以用来显示View.官方的说法是Activity一个应用程序的组件,它提供一个屏幕来与用户交互,以便做一些诸如打电话. ...

  9. centos7安装mysql 8

    1.查看已有mysql(包括mariadb) rpm -pa | grep mariadb yum list installed | grep mariadb rpm -pa | grep mysql ...

  10. CSS3小清新下拉菜单 简易大方

    之前有分享过几款CSS3菜单和jQuery菜单,像这款HTML5/CSS3自定义下拉框 3D卡片折叠动画3D效果非常华丽,这次要分享的这款相对比较简单,很适合用在用户的操作面板上.先来看看效果图: 怎 ...