news 2026/7/5 5:11:52

Reader的源码、FilterReader源码、PushbackReader源码(windows操作系统,JDK8)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Reader的源码、FilterReader源码、PushbackReader源码(windows操作系统,JDK8)
、Reader.class源码

Reader 是用来读取字符流的装饰器模式中顶层的抽象类,与 InputStream(字节流)不同的是,Reader 专门处理字符char(字符char在JVM中使用Unicode编码占2个byte),主要用于读取和写入中文文本。
windows操作系统的JDK8版本中,所有的Reader的子类如下(此处只展示部分):

常用的Reader子类有以下5个:

①、InputStreamReader:处理文件的字符输入流,在进行文件读操作时,如果遇到不同编码格式,可以使用 InputStreamReader 进行处理。
②、BufferedReader :类似于 BufferedInputStream ,不同点在于BufferedReader 读取的是字符char,BufferedInputStream读取的是字节byte,其内部也带有一个缓冲区(1个char[]数组),一般用BufferedReader来配合其它字符输入流一起使用来减少IO次数,也可以使用mark()函数和 reset()函数重复从缓冲区(1个char[]数组)中读取字符。
③、PushbackReader:以将已读取的字符(char)重新推回输入流中。这样,下次调用read()函数时,这些字符(char)就会再次被读取。因此,PushbackReader可以从输入流中解析字符(char)数据。
④、StringReader:如果输入源是一个字符串的,可以使用 StringReader 来将一个字符串转换为字符流。
以上4个子类的使用方式请参照:
1、Java的IO概览(一)
2、Java的IO概览(二)

Reader.class的源码如下:

package java.io; public abstract class Reader implements Readable, Closeable { //当这个字符输入流进行read操作、skip操作的时候,用这个对象(锁)来同步线程,来保证多线程场景下操作这个字符输入流时的线程安全性。 //这个锁对象既可以是Reader.class类型对象本身,也可以是其它类型的对象,但是最有效的同步方式是,在使用synchronized锁定同步代码块的时候,一定要用lock变量指向的锁对象,而不是将synchronized加到函数上面或者用this关键字同步,推荐的用法详情请看skip()函数 //推荐的同步方式如下: /** * public long skip(long n) throws IOException { * ...省略部分代码... * synchronized (lock) { * if ((skipBuffer == null) || (skipBuffer.length < nn)) * skipBuffer = new char[nn]; * long r = n; * while (r > 0) { * int nc = read(skipBuffer, 0, (int)Math.min(r, nn)); * if (nc == -1) * break; * r -= nc; * } * return n - r; * } *} * */ //以下2种方式不推荐 //不推荐的同步方式①:将synchronized关键字加到函数上面 /** * public synchronized long skip(long n) throws IOException { * ...省略部分代码... * */ //不推荐的同步方式②:将lock改成this关键字 /** * public long skip(long n) throws IOException { * ...省略部分代码... * synchronized (this) { * if ((skipBuffer == null) || (skipBuffer.length < nn)) * skipBuffer = new char[nn]; * long r = n; * while (r > 0) { * int nc = read(skipBuffer, 0, (int)Math.min(r, nn)); * if (nc == -1) * break; * r -= nc; * } * return n - r; * } *} * */ protected Object lock; //构造函数,并将用于多线程场景下线程同步的lock变量指向this(当前这个字符输入流对象)本身 protected Reader() { this.lock = this; } //构造函数,并将用于多线程场景下线程同步的lock变量指向其它对象而不是this(当前这个字符输入流对象)本身 protected Reader(Object lock) { if (lock == null) { throw new NullPointerException(); } this.lock = lock; } //将从字符输入流中读取n个字符放入到NIO中的CharBuffer对象(字符缓冲区对象)里面 //当CharBuffer对象(字符缓冲区对象)中可用的空间>0时,这个函数是阻塞的,直到读取到数据或者抛出异常。 public int read(java.nio.CharBuffer target) throws IOException { int len = target.remaining();//计算NIO中的CharBuffer对象(字符缓冲区对象)中还可以放多少个字符(可用的空间) char[] cbuf = new char[len];//构建一个长度为len的字符数组char[],len表示CharBuffer对象(字符缓冲区对象)中还可以放的字符数量 int n = read(cbuf, 0, len);//从字符输入流中读取len个字符放到字符数组char[] cbuf的[0,len)索引位置,并返回从字符输入流中读取到的字符数量 if (n > 0) //如果从字符输入流中读取到的字符数量>0,则将这些字符(假设有n个字符,0<n<=len)都放入到CharBuffer对象(字符缓冲区对象)中 target.put(cbuf, 0, n); return n;//返回n,这里的n取值范围为0<=n<=len,或者[0,len] } //将从字符输入流中读取1个字符 //和InputStream一样,在读取到数据或者抛出异常前,这个函数是阻塞的。 public int read() throws IOException { char cb[] = new char[1];//构建一个长度为1的字符数组char[] cb //将从字符输入流中读取1个字符放入到字符数组char[] cb的[0,1)索引位置,如果字符输入流已经读完了(即已经读取到了文件的EOF符号),则会返回-1,read(cb, 0, 1)函数是留给子类实现的 if (read(cb, 0, 1) == -1) return -1;//如果字符输入流已经读完了(即已经读取到了文件的EOF符号),则返回-1 else return cb[0];//否则,返回读取到的字符 } //将从字符输入流中读取length个字符到字符数组char[] cbuf中,length表示字符数组char[] cbuf的长度 //和InputStream一样,在读取到数据或者抛出异常前,这个函数是阻塞的。 public int read(char cbuf[]) throws IOException { return read(cbuf, 0, cbuf.length); } //留给子类实现,如果读到了字符输入流的末尾(即已经读取到了文件的EOF符号)则会返回-1 abstract public int read(char cbuf[], int off, int len) throws IOException; //每一次从字符输入流中最多跳过的字符数量 private static final int maxSkipBufferSize = 8192; //执行skip()函数时才会用到,用来跳过指定的字符数量时需要借助字符数组char[]来完成 private char skipBuffer[] = null; //将从字符输入流中跳过n个字符 public long skip(long n) throws IOException { if (n < 0L) //校验,如果n<0,则抛出一个IllegalArgumentException throw new IllegalArgumentException("skip value is negative"); //每次跳过的字符最多为8192个 int nn = (int) Math.min(n, maxSkipBufferSize); synchronized (lock) {//后续的代码片段只允许单线程执行 if ((skipBuffer == null) || (skipBuffer.length < nn)) skipBuffer = new char[nn];//用于每次跳过指定数量字符(每次最多为8192个)的字符数组 long r = n;//r表示当前线程还没(或者还需要)跳过的字符的总数量 while (r > 0) { //当前线程还没(或者还需要)跳过的字符的总数量>0时,每次执行read()函数从字符输入流中跳过0~8192个字符 //当前线程读到了字符输入流的末尾(即已经读取到了文件的EOF符号)则会返回-1 int nc = read(skipBuffer, 0, (int)Math.min(r, nn)); if (nc == -1) break;//中断while循环 r -= nc;//将本次从字符输入流中跳过的字符递减就是接下来还没(或者还需要)跳过的字符的总数量 } return n - r;//返回已经从字符输入流中跳过的字符总数量(n-r) } } //具体逻辑需要子类来重写,JDK给这个函数的定义如下: //①、如果返回 true,则表示下一次调用read()函数时保证不会阻塞,否则返回 false。 //②、即使返回 false 也不能保证下一次调用read()函数时一定会阻塞。 public boolean ready() throws IOException { return false; } //具体逻辑需要子类来重写,JDK给这个函数的定义如下: //①、如果返回true,表示子类支持mark()函数; //②、如果返回false,表示子类不支持mark()函数 public boolean markSupported() { return false; } //标记此字符输入流中的当前位置。随后调用reset()函数时会将此字符输入流重新定位到上次标记的位置,从而使得后续的读取操作能够再次读取相同的字符。 //ASCIIReader、BufferedReader、CharArrayReader、FilterReader、Latin1Reader、LineNumberReader、StringReader、UCSReader会重写这个mark()函数,其余Reader.class的子类不会重写这个函数。 public void mark(int readAheadLimit) throws IOException { throw new IOException("mark() not supported"); } //reset()函数会将此字符输入流重新定位到上次执行mark()函数标记的位置,从而使得后续的读取操作能够再次读取相同的字符。 //ASCIIReader、BufferedReader、CharArrayReader、FilterReader、Latin1Reader、LineNumberReader、StringReader、UCSReader会重写这个reset()函数,其余Reader.class的子类不会重写这个函数。 public void reset() throws IOException { throw new IOException("reset() not supported"); } /** * Closes the stream and releases any system resources associated with * it. Once the stream has been closed, further read(), ready(), * mark(), reset(), or skip() invocations will throw an IOException. * Closing a previously closed stream has no effect. * * @exception IOException If an I/O error occurs */ //留给子类实现,子类必须遵守以下规则: //①、关闭这个字符输入流,并释放与该流相关的系统资源 //②、关闭的字符输入流对象在执行read()函数, ready()函数, mark()函数, reset()函数, 或者 skip()函数 时将抛出一个 IOException. abstract public void close() throws IOException; }
1.1、Reader的skip()函数
//将从字符输入流中跳过n个字符 public long skip(long n) throws IOException { if (n < 0L) //校验,如果n<0,则抛出一个IllegalArgumentException throw new IllegalArgumentException("skip value is negative"); //每次跳过的字符最多为8192个 int nn = (int) Math.min(n, maxSkipBufferSize); synchronized (lock) {//后续的代码片段只允许单线程执行 if ((skipBuffer == null) || (skipBuffer.length < nn)) skipBuffer = new char[nn];//用于每次跳过指定数量字符(每次最多为8192个)的字符数组 long r = n;//r表示当前线程还没(或者还需要)跳过的字符的总数量 while (r > 0) { //当前线程还没(或者还需要)跳过的字符的总数量>0时,每次执行read()函数从字符输入流中跳过0~8192个字符 //当前线程读到了字符输入流的末尾(即已经读取到了文件的EOF符号)则会返回-1 int nc = read(skipBuffer, 0, (int)Math.min(r, nn)); if (nc == -1) break;//中断while循环 r -= nc;//将本次从字符输入流中跳过的字符递减就是接下来还没(或者还需要)跳过的字符的总数量 } return n - r;//返回已经从字符输入流中跳过的字符总数量(n-r) } }

如果一个字符输入流中有20000个字符,并且此时有多个线程(此处假设有2个线程,分别为ThreadA和ThreadB)要执行此字符输入流的skip()函数,ThreadA执行了skip(8000)表示要从这个字符输入流中跳过8000个字符,ThreadB执行了skip(10000)表示要从这个字符输入流中跳过10000个字符,此时,这2个线程同时执行skip()函数的过程如下所示:
①、ThreadA和ThreadB分别在各自的线程栈中压入skip()函数,如下所示:

②、假如ThreadA在执行下面的代码时先获取到了锁,先进入到了Monitor监控(synchronized底层是基于操作系统的Monitor来实现的)的代码片段

synchronized (lock) {//后续的代码片段只允许单线程执行

那么ThreadA在进入while循环之前,会进行初始化char[]数组的长度为8000和临时变量long r的操作,如下所示:

③、ThreadA只会进入一次while循环,ThreadA在进入while循环之后,会从字符输入流中读取8000个字符到字符数组char[] skipBuffer中,然后就结束while循环,并且释放掉获取到的锁对象Object lock,并且将从字符输入流中跳过的字符总数量(这里是8000)返回给该函数的调用者,如下所示:

④、然后ThreadB在执行下面的代码时也获取到了锁,进入到了Monitor监控(synchronized底层是基于操作系统的Monitor来实现的)的代码片段

synchronized (lock) {//后续的代码片段只允许单线程执行

那么ThreadB在进入while循环之前,会重新初始化char[]数组的长度为8192和临时变量long r的操作,如下所示:

⑤、ThreadB第1次进入while循环之后,会从字符输入流中读取8192个字符到字符数组char[] skipBuffer中,然后更新long r = 1808,如下所示:

⑥、ThreadB第2次进入while循环之后,会从字符输入流中读取1808个字符到字符数组char[] skipBuffer中,然后更新long r = 0,如下所示:

⑦、最后,ThreadB结束while循环,并且释放掉获取到的锁对象Object lock,并且将从字符输入流中跳过的字符总数量(这里是10000)返回给该函数的调用者,如下所示:

二、FilterReader.class源码——字符流的装饰器基类

FilterReader 的UML关系图,如下所示:

FilterReader.class的源码,如下所示:

package java.io; /** * Abstract class for reading filtered character streams. * The abstract class <code>FilterReader</code> itself * provides default methods that pass all requests to * the contained stream. Subclasses of <code>FilterReader</code> * should override some of these methods and may also provide * additional methods and fields. * * @author Mark Reinhold * @since JDK1.1 */ public abstract class FilterReader extends Reader { //实际被装饰的字符输入流 protected Reader in; //当用构造函数创建这个装饰器时,传入一个被装饰者的字符输入流 protected FilterReader(Reader in) { super(in); this.in = in; } //调用实际被装饰的字符输入流的read() 函数 public int read() throws IOException { return in.read(); } //调用实际被装饰的字符输入流的read(char cbuf[], int off, int len) 函数 public int read(char cbuf[], int off, int len) throws IOException { return in.read(cbuf, off, len); } //调用实际被装饰的字符输入流的skip(long n) 函数 public long skip(long n) throws IOException { return in.skip(n); } //调用实际被装饰的字符输入流的ready() 函数 public boolean ready() throws IOException { return in.ready(); } //调用实际被装饰的字符输入流的markSupported() 函数 public boolean markSupported() { return in.markSupported(); } //调用实际被装饰的字符输入流的mark(int readAheadLimit) 函数 public void mark(int readAheadLimit) throws IOException { in.mark(readAheadLimit); } //调用实际被装饰的字符输入流的reset() 函数 public void reset() throws IOException { in.reset(); } //调用实际被装饰的字符输入流的close() 函数 public void close() throws IOException { in.close(); } }
三、PushbackReader.class源码——可以回退字符到被读取的字符流中的字符流装饰器类

PushbackReader与PushbackInputStream类似,只不过PushbackReader提供了有限字符(内部定义了一个默认长度为1的char[] buf字符数组)的缓冲式回退能力,具体过程如下:
①、当调用unread()函数时会将任意字符(可以是从被装饰的输入流中读取的字节,也可以是自己定义的字符)压入char[] buf字符数组的尾部(pos-1的位置);
②、当后续调用read()函数时优先读取步骤①中这个char[] buf字符数组中被压入的字符。
PushbackInputStream.class的相关源码和使用方式请参照:13、PushbackInputStream和StreamTokenizer的源码分析和使用方法详细分析

3.1、PushbackReader.class的源码分析

PushbackReader.class 的UML关系图,如下所示:

PushbackReader.class 的源码,如下所示:

package java.io; public class PushbackReader extends FilterReader { //有限长度的用于回退的字符数组缓冲区,默认长度为1 private char[] buf; //可读指针,char[] buf(有限长度的用于回退的字符数组缓冲区)中该指针(包括该指针)索引之后的所有字符都可以读 private int pos; //构造函数,in为被装饰的字符输入流,size为char[] buf(有限长度的用于回退的字符数组缓冲区)的长度 public PushbackReader(Reader in, int size) { super(in); if (size <= 0) { throw new IllegalArgumentException("size <= 0"); } this.buf = new char[size]; this.pos = size;//将可读指针指向char[] buf(有限长度的用于回退的字符数组缓冲区)中最后一个索引(size-1)之后 } //构造函数,in为被装饰的字符输入流 public PushbackReader(Reader in) { this(in, 1);//构造一个默认长度为1的char[] buf(用于回退的字符数组缓冲区) } //检查被装饰的字符输入流是否关闭 private void ensureOpen() throws IOException { if (buf == null) throw new IOException("Stream closed"); } //这个函数在多线程的场景下运行时是线程安全的 //如果char[] buf(用于回退的字符数组缓冲区)中有可读的字符的话,就从该缓冲区中读取1个字符 //如果char[] buf(用于回退的字符数组缓冲区)中没有可读的字符的话,就从被装饰的输入流中读取1个字符 //如果char[] buf(用于回退的字符数组缓冲区)和被装饰的输入流中都没有可读的字符的话,返回-1 public int read() throws IOException { synchronized (lock) {//用Reader.class::lock变量指向的对象(锁)来同步线程 ensureOpen(); if (pos < buf.length) return buf[pos++]; else return super.read(); } } //这个函数在多线程的场景下运行时是线程安全的,尽可能的从char[] buf(用于回退的字符数组缓冲区)和被装饰的输入流中读取len个字符到char[] cbuf的[off,off+len)索引位置,总共分为以下5种场景: //①、如果char[] buf(用于回退的字符数组缓冲区)中有len个字符的话,就从该缓冲区中读取len个字符到字符数组char[] cbuf的[off,off+len)索引位置 //②、如果char[] buf(用于回退的字符数组缓冲区)中没有任何字符并且被装饰的字符输入流中有len个字符,那就从被装饰的字符输入流中读取len个字符到字符数组char[] cbuf的[off,off+len)索引位置 //③、如果char[] buf(用于回退的字符数组缓冲区)中没有任何字符并且被装饰的字符输入流中只有avail(avail<len)个字符,那就从被装饰的字符输入流中读取avail个字符到字符数组char[] cbuf的[off,off+avail)索引位置 //④、如果char[] buf(用于回退的字符数组缓冲区)中有avail(avail<len)个字符的话,就读取avail个字符,剩余len-avail个字符从被装饰的字符输入流中读取,如果被装饰的字符输入流中没有len-avail个字符的话,那就从被装饰的输入流中有多少读取多少,直到将被装饰的输入流读取完毕,然后将以上2个地方(被装饰的输入流+用于回退的字符数组缓冲区)读取的所有字符(假如有x个)放入到字符数组char[] cbuf的[off,off+x)索引位置 //⑤、如果char[] buf(用于回退的字符数组缓冲区)和被装饰的字符输入流中都没有任何字符的话,返回-1 public int read(char cbuf[], int off, int len) throws IOException { synchronized (lock) { //检查被装饰的字符输入流是否关闭 ensureOpen(); try { if (len <= 0) { if (len < 0) { throw new IndexOutOfBoundsException(); } else if ((off < 0) || (off > cbuf.length)) {//相当于off + len > cbuf.length(源码中这样写代码的好处我没看出来) throw new IndexOutOfBoundsException(); } return 0;//要从PushbackReader 对象中读取的len个字符==0时,返回0 } int avail = buf.length - pos;//用于回退的字符数组缓冲区中实际装载了buf.length - pos个字符 if (avail > 0) { if (len < avail) avail = len; System.arraycopy(buf, pos, cbuf, off, avail); pos += avail; off += avail; len -= avail; } if (len > 0) { len = super.read(cbuf, off, len); if (len == -1) { return (avail == 0) ? -1 : avail; } return avail + len;//场景④中的x就是这里的avail + len } return avail; } catch (ArrayIndexOutOfBoundsException e) { throw new IndexOutOfBoundsException(); } } } //一次只可以回推1个字符数据到char[] buf(用于回退的字符数组缓冲区)中 public void unread(int c) throws IOException { synchronized (lock) { ensureOpen(); if (pos == 0)//pos=0时,表示char[] buf(用于回退的字符数组缓冲区)中已经没有足够的容量再放置数据,所以抛出一个IOException异常。 throw new IOException("Pushback buffer overflow"); buf[--pos] = (char) c; } } //一次回推char[] cbuf字符数组中[off,off+len)索引位置的len个字符数据到char[] buf(用于回退的字符数组缓冲区)中 public void unread(char cbuf[], int off, int len) throws IOException { synchronized (lock) { ensureOpen(); if (len > pos)//如果char[] buf(用于回退的字符数组缓冲区)中没有足够的位置放置len个字符,则抛出一个IOException throw new IOException("Pushback buffer overflow"); pos -= len;//如果char[] buf(用于回退的字符数组缓冲区)中有足够的位置放置len个字符,则使用System.arraycopy()函数进行回退 System.arraycopy(cbuf, off, buf, pos, len); } } public void unread(char cbuf[]) throws IOException { unread(cbuf, 0, cbuf.length); } public boolean ready() throws IOException { synchronized (lock) { ensureOpen(); return (pos < buf.length) || super.ready(); } } //不支持mark()函数 public void mark(int readAheadLimit) throws IOException { throw new IOException("mark/reset not supported"); } //不支持reset()函数 public void reset() throws IOException { throw new IOException("mark/reset not supported"); } //不支持mark()函数 public boolean markSupported() { return false; } //关闭被装饰的字符输入流和用于回退的字符数组缓冲区 public void close() throws IOException { super.close(); buf = null; } //从char[] buf(用于回退的字符数组缓冲区)+被装饰的字符输入流中跳过n个字符,如果char[] buf(用于回退的字符数组缓冲区)+被装饰的字符输入流中的字符数量<n,则返回实际跳过的字符数量 public long skip(long n) throws IOException { if (n < 0L) throw new IllegalArgumentException("skip value is negative"); synchronized (lock) { ensureOpen(); int avail = buf.length - pos;//从char[] buf(用于回退的字符数组缓冲区)中跳过的字符 if (avail > 0) { if (n <= avail) { pos += n; return n; } else { pos = buf.length; n -= avail; } } return avail + super.skip(n);//从被装饰的字符输入流中跳过的字符累加到从char[] buf(用于回退的字符数组缓冲区)中跳过的字符 } } }
3.2、PushbackReader.class的read()函数和unread()函数
package java.io; public class PushbackReader extends FilterReader { //有限长度的用于回退的字符数组缓冲区,默认长度为1 private char[] buf; //可读指针,char[] buf(有限长度的用于回退的字符数组缓冲区)中该指针(包括该指针)索引之后的所有字符都可以读 private int pos; ...省略部分代码... //这个函数在多线程的场景下运行时是线程安全的,尽可能的从char[] buf(用于回退的字符数组缓冲区)和被装饰的输入流中读取len个字符到char[] cbuf的[off,off+len)索引位置,总共分为以下5种场景: //①、如果char[] buf(用于回退的字符数组缓冲区)中有len个字符的话,就从该缓冲区中读取len个字符到字符数组char[] cbuf的[off,off+len)索引位置 //②、如果char[] buf(用于回退的字符数组缓冲区)中没有任何字符并且被装饰的字符输入流中有len个字符,那就从被装饰的字符输入流中读取len个字符到字符数组char[] cbuf的[off,off+len)索引位置 //③、如果char[] buf(用于回退的字符数组缓冲区)中没有任何字符并且被装饰的字符输入流中只有avail(avail<len)个字符,那就从被装饰的字符输入流中读取avail个字符到字符数组char[] cbuf的[off,off+avail)索引位置 //④、如果char[] buf(用于回退的字符数组缓冲区)中有avail(avail<len)个字符的话,就读取avail个字符,剩余len-avail个字符从被装饰的字符输入流中读取,如果被装饰的字符输入流中没有len-avail个字符的话,那就从被装饰的输入流中有多少读取多少,直到将被装饰的输入流读取完毕,然后将以上2个地方(被装饰的输入流+用于回退的字符数组缓冲区)读取的所有字符(假如有x个)放入到字符数组char[] cbuf的[off,off+x)索引位置 //⑤、如果char[] buf(用于回退的字符数组缓冲区)和被装饰的字符输入流中都没有任何字符的话,返回-1 public int read(char cbuf[], int off, int len) throws IOException { synchronized (lock) { //检查被装饰的字符输入流是否关闭 ensureOpen(); try { if (len <= 0) { if (len < 0) { throw new IndexOutOfBoundsException(); } else if ((off < 0) || (off > cbuf.length)) {//相当于off + len > cbuf.length(源码中这样写代码的好处我没看出来) throw new IndexOutOfBoundsException(); } return 0;//要从PushbackReader 对象中读取的len个字符==0时,返回0 } int avail = buf.length - pos;//用于回退的字符数组缓冲区中实际装载了buf.length - pos个字符 if (avail > 0) { if (len < avail) avail = len; System.arraycopy(buf, pos, cbuf, off, avail); pos += avail; off += avail; len -= avail; } if (len > 0) { len = super.read(cbuf, off, len); if (len == -1) { return (avail == 0) ? -1 : avail; } return avail + len;//场景④中的x就是这里的avail + len } return avail; } catch (ArrayIndexOutOfBoundsException e) { throw new IndexOutOfBoundsException(); } } } //一次只可以回推1个字符数据到char[] buf(用于回退的字符数组缓冲区)中 public void unread(int c) throws IOException { synchronized (lock) { ensureOpen(); if (pos == 0)//pos=0时,表示char[] buf(用于回退的字符数组缓冲区)中已经没有足够的容量再放置数据,所以抛出一个IOException异常。 throw new IOException("Pushback buffer overflow"); buf[--pos] = (char) c; } } ...省略部分代码... }

如果使用者使用的被装饰的字符输入流是CharArrayReader,然后执行PushbackReader.class的read()函数和unread()函数时,使用如下代码:

package com.xxx.bio; import java.io.CharArrayReader; import java.io.IOException; import java.io.PushbackReader; import java.io.Reader; public class PushbackReaderTest { public static void main(String[] args) throws IOException { String str = "你好,世界,我将征服你"; Reader reader = new CharArrayReader(str.toCharArray()); //构建字符回退流 PushbackReader pushbackReader = new PushbackReader(reader, 8); int len = -1; System.out.println("输出内容:"); while ((len = pushbackReader.read()) != -1) { //转为char类型 char c = (char) len; if (c == ',') {//此处为中文"," //为,号时 ,先往前读3个,再往后倒两个 char[] b1 = new char[3]; pushbackReader.read(b1); //往后倒两个 pushbackReader.unread(b1, 0, 2); } else { System.out.print(c); } } } }

上面代码的执行结果如下:

以上代码的整个执行过程分为以下5步:
①、通过构造函数构建一个长度为8的char[] buf(用于回退的字符数组缓冲区)和CharArrayReader.class类型的被装饰的字符输入流,如下所示:

PushbackReader pushbackReader = new PushbackReader(reader, 8);

②、按照顺序从CharArrayReader.class类型的输入流中读取字符,直到读取到中文“,”时,如下所示:

while ((len = pushbackReader.read()) != -1) { //转为char类型 char c = (char) len; if (c == ',') {//此处为中文"," } else { System.out.print(c); } }

输出如下:

你好

③、当第一次读取到中文“,”之后,从被装饰的字符输入流CharArrayReader.class中往char[] b1字符数组中读取3个字符,如下所示:

//为,号时 ,先往前读3个,再往后倒两个 char[] b1 = new char[3]; pushbackReader.read(b1);

④、将步骤③中读入到char[] b1字符数组中的[0,2)索引位置的数据读取到PushbackReader中的char[] buf(用于回退的字符数组缓冲区)的[6,8)索引位置中,如下所示:

//往后倒两个 pushbackReader.unread(b1, 0, 2);

⑤、再次重复执行步骤②中按照顺序从CharArrayReader.class类型的字符输入流中读取字符时,先读取PushbackReader中的char[] buf(用于回退的字符数组缓冲区)的[6,8)索引位置,再从CharArrayReader.class类型的字符输入流中读取剩余字符,如下所示:

while ((len = pushbackReader.read()) != -1) { //转为char类型 char c = (char) len; if (c == ',') {//此处为中文"," } else { System.out.print(c); } }

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/7/5 5:11:11

嵌入式Linux应用开发,到底和桌面开发差在哪

你有没有过这种经历——在Ubuntu上写了个C程序&#xff0c;gcc编译通过&#xff0c;跑起来一切正常&#xff0c;然后丢到ARM开发板上&#xff0c;./a.out敲下去&#xff0c;系统回你一句"cannot execute binary file"&#xff1f;道理很简单。x86_64编译出来的东西&a…

作者头像 李华
网站建设 2026/7/5 5:10:22

终极免费岛屿设计工具:Happy Island Designer 快速入门指南

终极免费岛屿设计工具&#xff1a;Happy Island Designer 快速入门指南 【免费下载链接】HappyIslandDesigner "Happy Island Designer (Alpha)"&#xff0c;是一个在线工具&#xff0c;它允许用户设计和定制自己的岛屿。这个工具是受游戏《动物森友会》(Animal Cros…

作者头像 李华
网站建设 2026/7/5 5:09:13

网络计划WebApp求解:融合Python与AI决策的项目管理系统

一、平台总体架构&#xff1a;项目调度的数字实验室网络计划问题本质上是一个具有时间约束、逻辑约束与资源约束的复杂网络系统。项目中的各项工作并不是孤立存在&#xff0c;而是通过先后关系、持续时间和资源需求相互连接&#xff0c;共同决定整个项目的执行效率与最终工期。…

作者头像 李华
网站建设 2026/7/5 5:06:37

3步快速部署Google Cloud Vision API示例项目

3步快速部署Google Cloud Vision API示例项目 【免费下载链接】cloud-vision Sample code for Google Cloud Vision 项目地址: https://gitcode.com/gh_mirrors/cl/cloud-vision Google Cloud Vision API是一个强大的图像识别和分析服务&#xff0c;提供了人脸检测、标签…

作者头像 李华