Java IO流之StringWriter和StringReader用法分析

简介

StringWriter和StringReader分别继承自Writer和Reader抽象类,作用就是将字符串String类型适配到StringWriter和StringReader类.

其中StringWriter实际上是用StringBuffer实现的,StringBuffer是一个线程安全的类,实际上默认的是16个字符的char数组.所以通过方法write()和append()等其他重载的方法.将字符串添加到StringBuffer中,可以调用toString()或者getBuffer()方法显示StringWriter流中所有的数据.其中close()方法和flush()方法,无实际的功能,只是简单的继承.

StringReader的有参构造方法传入的是String类型的数据,通过read()重载方法读取流中部分或者全部的字符内容.

StringWriter介绍

1.构造方法

public StringWriter() {}
public StringWriter(int initialSize) {}

无参构造方法,创建的是默认大小为StringBuffer的StringWriter流,其中StringBuffer底层实际是一个字符数组,默认大小是16个字符.

有参构造方法,创建的是指定大小的StringBuffer的StringWriter流,也就是指定initialSize个字符的字符数组.

2.内部变量

private StringBuffer buf;

StringWriter流中缓冲区实际上通过StringBuffer来实现的.

3. 内部方法

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() {}
  • write(int c)---将单个字符写到流中.
  • write(char cbuf[] ,int off,int len)---将字符数组cbuf中off开始,len个字符写到流中.
  • write(String str)---将字符串str写到流中.
  • write(String str,int off,int len)---将字符串str中off位置开始,len个字符写到流中.
  • append(charsequence csq)---将字符序列csq写到流中.
  • append(charsequence csq,int start ,int end)----将字符序列中start到end之间字符写到流中.
  • append(char c)---将单个字符写到流中.
  • toString()---将流中所有字符转换成字符串返回.
  • getBuffer()----将流中内容,实际是StringBuffer返回.
  • flush()---仅仅是继承,方法内没有任何实现.
  • close()----仅仅是继承,方法内没有任何实现.

StringReader介绍

1.构造方法

public StringReader(String s) {}

有参构造方法,传入的实际上是流中内容为字符串s.

2.内部变量

  private String str;
  private int length;
  private int next = 0;
  private int mark = 0;
  • str---StringReader流中实际的数据就是通过构造方法传入的字符串str.
  • length---流中字符串的长度.
  • next---要读取的下一个字符的位置.
  • mark---标记字符位置,调用mark()方法会将当前位置保存到mark中.

3. 内部方法

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(){}
  • ensureOpen()---确保流没有关闭,实际判断是流中内容字符串str不为null.
  • read()---从流中读取一个字符.
  • read(char cbuf[],int off,int len)---将流中最多len个字符读取到字符数组cbuf中,cbuf中从off开始.
  • skip(long ns)---跳过ns个字符.
  • ready()---流是否准备读取.
  • markSupported()---是否支持标记
  • mark(int readAheadLimit)---标记当前位置,readAheadLimit表示的是在标记有效情况下,从当前位置开始可取的最大字符数.
  • reset()---将流中当前位置重置到标记位置.
  • close()---关闭流,实际上是将流中内容字符串str置为null.

案例

public class StringWriterDemo {
  public static void main(String[] args) throws IOException {
    testStringWriter();
    testStringReader();
    testOfAnotherStringWriter();
  }
  //测试StringWriter
  private static void testStringWriter() throws IOException {
    StringWriter sw = new StringWriter();
    sw.append("this is a");
    sw.write("demo of StringWriter");
    System.out.println(sw.toString());
    sw.close();
  }

  //测试StringReader
  private static void testStringReader() throws IOException {
   String content = "this is a demo of StringReader";
   StringReader sr = new StringReader(content);
   int c;
   while((c=sr.read())!=-1) {
     System.out.println((char)c);
   }
    sr.close();
  }

  //将读取的数据先写到StringWriter流中,然后直接调用toString()方法打印结果
  private static void testOfAnotherStringWriter() throws IOException {
    FileReader fr = new FileReader("D:\\java.txt");
    BufferedReader br = new BufferedReader(fr);
    StringWriter sw = new StringWriter();
    int n;
    char cbuf[] = new char[1024];
    while((n=br.read(cbuf))!=-1) {
      sw.write(cbuf, 0, n);
    }
    System.out.println(sw.toString());
    fr.close();
    br.close();
  }
}

源码分析

1.StringWriter源码分析

public class StringWriter extends Writer {
    //StringWriter中缓冲功能是通过StringBuffer来实现的.
    private StringBuffer buf;

    //创建默认缓冲为16个字符的StringWriter.默认的StringBuffer实际是大小为16个字符的字符数组.
    public StringWriter() {
        buf = new StringBuffer();
        lock = buf;
    }

    //创建指定大小StringBuffer的StringWriter,也就是指定了initialSize个字符的字符数组.
    public StringWriter(int initialSize) {
        if (initialSize < 0) {
            throw new IllegalArgumentException("Negative buffer size");
        }
        buf = new StringBuffer(initialSize);
        lock = buf;
    }

    //向流写入单个字符c
    public void write(int c) {
        buf.append((char) c);
    }

    //将字符数组cbuf中off位置开始,len个字符写到流中
    public void write(char cbuf[], int off, int len) {
        if ((off < 0) || (off > cbuf.length) || (len < 0) ||
            ((off + len) > cbuf.length) || ((off + len) < 0)) {
            throw new IndexOutOfBoundsException();
        } else if (len == 0) {
            return;
        }
        buf.append(cbuf, off, len);
    }

    //将字符串str写到流中
    public void write(String str) {
        buf.append(str);
    }

    //将字符串str中off位置开始,len个字符写到流中
    public void write(String str, int off, int len)  {
        buf.append(str.substring(off, off + len));
    }

    //将字符序列csq添加到流中.
    public StringWriter append(CharSequence csq) {
        if (csq == null)
            write("null");
        else
            write(csq.toString());
        return this;
    }

    //将字符序列csq中start位置到end之间的字符添加到流中
    public StringWriter append(CharSequence csq, int start, int end) {
        CharSequence cs = (csq == null ? "null" : csq);
        write(cs.subSequence(start, end).toString());
        return this;
    }

    //向流中添加一个字符c
    public StringWriter append(char c) {
        write(c);
        return this;
    }

    //将流中字符转换成字符串返回.
    public String toString() {
        return buf.toString();
    }

    //返回流的内容,即StringBuffer
    public StringBuffer getBuffer() {
        return buf;
    }

    //刷新流,没有具体实现.
    public void flush() {
    }

    //关闭流后读取数据不会抛出异常,因为没有具体实现
    public void close() throws IOException {
    }

}

2,StringReader源码分析

public class StringReader extends Reader {
    //流中实际数据就是通过构造方法传入str字符串.
    private String str;
    //字符串str的长度
    private int length;
    //下一个要读取字符的位置.
    private int next = 0;
    //标记的位置.
    private int mark = 0;

    //创建一个StringReader流,参数传入的是字符串s
    public StringReader(String s) {
        this.str = s;
        this.length = s.length();
    }

   //确保流没有关闭,实际判断的是StringReader构造方法是否有传入字符串.
    private void ensureOpen() throws IOException {
        if (str == null)
            throw new IOException("Stream closed");
    }

    //读取单个字符
    public int read() throws IOException {
        synchronized (lock) {
            ensureOpen();
            if (next >= length)
                return -1;
            return str.charAt(next++);
        }
    }

    //将缓冲区中最多len个字符写到字符数组cbuf中,位置从off开始,长度为len个
    public int read(char cbuf[], int off, int len) throws IOException {
        synchronized (lock) {
            ensureOpen();
            //检测参数的合法性
            if ((off < 0) || (off > cbuf.length) || (len < 0) ||
                ((off + len) > cbuf.length) || ((off + len) < 0)) {
                throw new IndexOutOfBoundsException();
            } else if (len == 0) {
                return 0;
            }
            //超过字符串的长度,返回-1
            if (next >= length)
                return -1;
            //剩余可读取的字符为length-next,
            //如果len超过剩余可读取字符,实际只能读取剩余可读取字符
            //如果len小于剩余可读取字符,实际上读取的是len个字符
            int n = Math.min(length - next, len);
            str.getChars(next, next + n, cbuf, off);
            next += n;
            return n;
        }
    }

    //跳过ns个字符,返回跳过的字符数
    public long skip(long ns) throws IOException {
        synchronized (lock) {
            ensureOpen();
            if (next >= length)
                return 0;
            // Bound skip by beginning and end of the source
            //缓冲区中剩余字符数与要跳过的字符数比较.取较小值
            long n = Math.min(length - next, ns);
            n = Math.max(-next, n);
            next += n;
            return n;
        }
    }

    //流是否准备读取.
    public boolean ready() throws IOException {
        synchronized (lock) {
        ensureOpen();
        return true;
        }
    }

    //流是否支持标记
    public boolean markSupported() {
        return true;
    }

   //标记缓冲区中当前位置,readAheadLimit表示的是在保留标记的情况下,最大可读取的字符数
    public void mark(int readAheadLimit) throws IOException {
        if (readAheadLimit < 0){
            throw new IllegalArgumentException("Read-ahead limit < 0");
        }
        synchronized (lock) {
            ensureOpen();
            mark = next;
        }
    }

    //将缓冲区当前位置重置到标记位置
    public void reset() throws IOException {
        synchronized (lock) {
            ensureOpen();
            next = mark;
        }
    }

    //关闭流,释放相关资源(实际只是将流中内容字符串置为空)
    public void close() {
        str = null;
    }
}
 

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • java基础知识之FileInputStream流的使用

    目录 一.File流概念 二.FileInputStream 1)FileInputStream概念 2)构造方法 3)FileInputStream常用API 三.三种read方法效率比较 一.File流概念 JAVA中针对文件的读写操作设置了一系列的流,其中主要有FileInputStream,FileOutputStream,FileReader,FileWriter四种最为常用的流 二.FileInputStream 1)FileInputStream概念 FileInputStream

  • 关于BufferedReader读取文件指定字符集问题

    目录 BufferedReader读取文件指定字符集问题 BufferedReader类读取中文乱码 BufferedReader读取文件指定字符集问题 默认的读取方式 BufferedReader bufferedReader = new BufferedReader(new FileReader(realPath)); 指定字符集方式 BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(new Fi

  • 聊聊为什么要使用BufferedReader读取File

    目录 使用BufferedReader读取File FileReader有三个用于read的函数 使用BufferedReader读取File的代码举例 使用BufferedReader循环读文件(新手要注意的细节) 1.BufferedReader读取一行 2.BufferedReader循环读每一行 3.注意细节 使用BufferedReader读取File Java编程思想一书里面有云,为了提高速度,可以用BufferedReader过滤器类,以提供缓冲区域.但是这个解释太简略了,看了这句

  • Java基础知识之StringReader流的使用

    目录 Java StringReader流的使用 一.StringReader流定义 二.StringReader的实例域 三.StringReader流构造函数 四.StringReader流的API 五.StringReader流的作用 StringReader分析 IO流分类图 StringReader Java StringReader流的使用 一.StringReader流定义 API说明:字符串输入流.其本质就是字符串 二.StringReader的实例域 //流对象 private

  • Java IO流之StringWriter和StringReader用法分析

    简介 StringWriter和StringReader分别继承自Writer和Reader抽象类,作用就是将字符串String类型适配到StringWriter和StringReader类. 其中StringWriter实际上是用StringBuffer实现的,StringBuffer是一个线程安全的类,实际上默认的是16个字符的char数组.所以通过方法write()和append()等其他重载的方法.将字符串添加到StringBuffer中,可以调用toString()或者getBuffe

  • Java字符流与字节流区别与用法分析

    本文实例讲述了Java字符流与字节流区别与用法.分享给大家供大家参考,具体如下: 字节流与字符流主要的区别是他们的的处理方式 流分类: 1.Java的字节流 InputStream是所有字节输入流的祖先,而OutputStream是所有字节输出流的祖先. 2.Java的字符流 Reader是所有读取字符串输入流的祖先,而writer是所有输出字符串的祖先. InputStream,OutputStream,Reader,writer都是抽象类.所以不能直接new 字节流是最基本的,所有的Inpu

  • Java IO流相关知识代码解析

    一.IO流的分类 字符流 Reader InputStreamReader(节点流) BufferedReader(处理流) Writer OutputStreamWriter(节点流) BufferedWriter(处理流) PrintWriter 字节流 InputStream FileInputStream(节点流) BufferedInputStream(处理流) ObjectInputStream(处理流) PrintStream OutputStream FileOutputStre

  • Java IO流之节点流与字符流的相关知识总结

    一.File file是文件和目录路径名的抽象表示 1.1 File的用法 用法: File file = new File("路径名"); //如 File file = new File("L:\\FileTestDemo\\AAA\\aaa.txt"); 注意:在windows中,路径名不能使用单个的\,单个的\为转义字符,可以使用\\,//或/ 1.2 File的常用方法 1.boolean createNewFile() 当且仅当具有此名称的文件尚不存在时

  • java IO流文件的读写具体实例

    引言: 关于java IO流的操作是非常常见的,基本上每个项目都会用到,每次遇到都是去网上找一找就行了,屡试不爽.上次突然一个同事问了我java文件的读取,我一下子就懵了第一反应就是去网上找,虽然也能找到,但自己总感觉不是很踏实,所以今天就抽空看了看java IO流的一些操作,感觉还是很有收获的,顺便总结些资料,方便以后进一步的学习... IO流的分类:1.根据流的数据对象来分:高端流:所有的内存中的流都是高端流,比如:InputStreamReader  低端流:所有的外界设备中的流都是低端流

  • Java IO流 文件的编码实例代码

    •文件的编码 package cn.test; import java.io.UnsupportedEncodingException; public class Demo15 { public static void main(String[] args) throws UnsupportedEncodingException { String str = "你好ABC123"; byte[] b1 = str.getBytes();//转换成字节系列用的是项目默认的编码 for (

  • 【Java IO流】字节流和字符流的实例讲解

    字节流和字符流 对于文件必然有读和写的操作,读和写就对应了输入和输出流,流又分成字节和字符流. 1.从对文件的操作来讲,有读和写的操作--也就是输入和输出. 2.从流的流向来讲,有输入和输出之分. 3.从流的内容来讲,有字节和字符之分. 这篇文章先后讲解IO流中的字节流和字符流的输入和输出操作. 一.字节流 1)输入和输出流 首先,字节流要进行读和写,也就是输入和输出,所以它有两个抽象的父类InputStream.OutputStream. InputStream抽象了应用程序读取数据的方式,即

  • Java IO流对象的序列化和反序列化实例详解

    Java-IO流 对象的序列化和反序列化 序列化的基本操作 1.对象序列化,就是将Object转换成byte序列,反之叫对象的反序列化. 2.序列化流(ObjectOutputStream),writeObject 方法用于将对象写入输出流中: 反序列化流(ObjectInputStream),readObject 方法用于从输入流中读取对象. 3.序列化接口(Serializeable) 对象必须实现序列化接口,才能进行序列化,否则会出现异常.这个接口没有任何方法,只是一个标准. packag

  • java IO流 之 输入流 InputString()的使用

    本文主要给大家介绍java的InputStream 流的使用. (1)FileInputstream: 子类,读取数据的通道 使用步骤: 1.获取目标文件:new File() 2.建立通道:new FileInputString() 3.读取数据:read() 4.释放资源:close() //一些默认要导入的包 import java.io.File; import java.io.FileInputStream; import java.io.IOException; public sta

  • Java IO流 File类的常用API实例

    •File类 1.只用于表示文件(目录)的信息(名称.大小等),不能用于文件内容的访问. package cn.test; import java.io.File; import java.io.IOException; public class Demo16 { public static void main(String[] args) { File file = new File("F:\\javaio"); //文件(目录)是否存在 if(!file.exists()) { /

随机推荐