我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Java基础知识之BufferedReader流的使用

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Java基础知识之BufferedReader流的使用

一、BufferedReader类概念

API文档描述:

BufferedReader类从字符输入流中读取文本并缓冲字符,以便有效地读取字符,数组和行

可以通过构造函数指定缓冲区大小也可以使用默认大小。对于大多数用途,默认值足够大

由Reader构成的每个读取请求都会导致相应的读取请求由基础字符或字节流构成,建议通过BufferedReader包装Reader的实例类以提高效率如


BufferedReader in  = new BufferedReader(new FileReader(“foo.in”));

使用DataInputStreams进行文本输入的程序可以通过用适当的BufferedReader替换每个DataInputStream来进行本地化

1)从字符输入流中读取文本并缓冲字符,以便有效地读取字符,数组和行怎么理解?

说明该类存在缓冲字符数组并且是该类可以高效读取字符的关键

2)构造函数指定缓冲区大小也可以使用默认大小怎么理解?

意味着该类存在的构造方法既可以传递数值指定缓冲区大小也可以由类中的默认大小指定

3)由Reader构成的每个读取请求都会导致相应的读取请求由基础字符或字节流构成,建议通过BufferedReader包装Reader的实例类以提高效率?

Reader构成的对象是字符对象,每次的读取请求都会涉及到字节读取解码字符的过程,而BufferedReader类中有设计减少这样的解码次数的方法,进而提高转换效率

4)BufferedReader替代DataInputStreams进行本地化?

需要查看DataInputStreams源码后才可知

二、BufferedReader类实例域


    // 字符输入流
    private Reader in; 
    // 字符缓冲区
    private char cb[]; 
    //读取字符存储的最末下标+1
    private int nChars; 
    //读取字符存储的起始下标
    private int nextChar; 
    private static final int INVALIDATED = -2;
    private static final int UNMARKED = -1;
    private int markedChar = UNMARKED;
 
    // 仅在markedChar为0时有效
    private int readAheadLimit = 0;
 
    // 如果下个字符是换行符,则跳过--专用于readLine()方法里面控制
    private boolean skipLF = false;
 
    // 设置标志时的markedSkipLF--用于mark()方法的变量
    private boolean markedSkipLF = false;
 
    // 默认的字符缓冲大小
    private static int defaultCharBufferSize =8192;
    
    //用于readLine()方法时初始化StringBuffer的初始容量
    private static int defaultExpectedLineLength = 80;

三、BufferedReader类构造函数

1)使用默认的缓冲区大小来创建缓冲字符输入流,默认大小为8192个字符


   private static int defaultCharBufferSize = 8192;  
   public BufferedReader(Reader in) {
        this(in, defaultCharBufferSize);
    } 
   public BufferedReader(Reader in, int sz) {
        super(in);
        if (sz <= 0)
            throw new IllegalArgumentException("Buffer size <= 0");
        this.in = in;
        cb = new char[sz];
        nextChar = nChars = 0;
    }

2)创建指定缓冲区大小的缓冲字符输入流,一般使用默认即可


  public BufferedReader(Reader in, int sz) {
        super(in);
        if (sz <= 0)
            throw new IllegalArgumentException("Buffer size <= 0");
        this.in = in;
        cb = new char[sz];
        nextChar = nChars = 0;
    }

四、BufferedReader类API

1)read()方法:读取1个或多个字节,返回一个字符,当读取到文件末尾时,返回-1


    
    public int read() throws IOException
    {
        synchronized (lock)
        {
            ensureOpen();
            for (;;)
            {
 
                 //一般条件为真,除非是使用了skip方法跳跃字节
                if (nextChar >= nChars)
                {
                    fill();  //调用该方法读取字符
 
                    if (nextChar >= nChars)
                        return -1;
                }
 
                 //此方法暂时不用管,涉及跳过字节数和换行符问题
                if (skipLF) 
                {
                    skipLF = false;
                    if (cb[nextChar] == '\n')
                    {
                        nextChar++;
                        continue;
                    }
                }
                return cb[nextChar++];
            }
        }
    }

实际流程图解:

2)fill()方法:从底层输入流中填充字符到缓冲区中,此方法会调用StreamDecoder的方法实现字节到字符的转换


    
    private void fill() throws IOException
    {
        int dst;
        
        //查看是否调用过make方法进行标记--若未使用make方法,则条件为真
        if (markedChar <= UNMARKED)
        {
          
            dst = 0;
        }
        else
        {
           
            int delta = nextChar - markedChar;
            if (delta >= readAheadLimit)
            {
             
                markedChar = INVALIDATED;
                readAheadLimit = 0;
                dst = 0;
            }
            else
            {
                if (readAheadLimit <= cb.length)
                {
              
                    System.arraycopy(cb, markedChar, cb, 0, delta);
                    markedChar = 0;
                    dst = delta;
                }
                else
                {
                   
                    char ncb[] = new char[readAheadLimit];
                    System.arraycopy(cb, markedChar, ncb, 0, delta);
                    cb = ncb;
                    markedChar = 0;
                    dst = delta;
                }
                nextChar = nChars = delta;
            }
        }
 
        int n;
        do
        {
            
          //调用InputStreamReader的方法,实际是调用StreamDecoder的read(char cbuf[], int offset, int length)方法   
            n = in.read(cb, dst, cb.length - dst);  
        }
        while (n == 0);
        if (n > 0)   //当读取到字符时
        {
            nChars = dst + n;   //字符缓冲区存储读到的字符的最末下标
            nextChar = dst;     //字符缓冲区存储读到的字符的起始下标
        }
    }

实际流程图解:注意根据read()方法先理解变量nChars和nextChar的意义

3)read(char cbuf[], int off, int len):将最多length个字符读入数组中,返回实际读入的字符个数,当读取到文件末尾时,返回-1,


    
    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;
            }
 
            int n = read1(cbuf, off, len); // 调用read1(cbuf, off, len)
            if (n <= 0)
                return n;
            while ((n < len) && in.ready())   // 注意该while循环,多次测试发现并未进入该方法,即使进入,本质还是调用read1(cbuf, off, len)
            {
                int n1 = read1(cbuf, off + n, len - n); 
                if (n1 <= 0)
                    break;
                n += n1;
            }
            return n;
        }
    }
 
 
 private int read1(char[] cbuf, int off, int len) throws IOException
    {
        if (nextChar >= nChars)
        {
 
            // 若请求的长度与缓冲区长度一样大时,直接会把字符读取到数组中,并未使用该类的字符缓冲区
 
            if (len >= cb.length && markedChar <= UNMARKED && !skipLF)
            {
                return in.read(cbuf, off, len);
            }
            fill();
        }
        if (nextChar >= nChars)
            return -1;
        if (skipLF) //若使用了换行、跳过字节数等才会考虑判断,暂时不用管
        {
            skipLF = false;
            if (cb[nextChar] == '\n')
            {
                nextChar++;
                if (nextChar >= nChars)
                    fill();
                if (nextChar >= nChars)
                    return -1;
            }
        }
        
        int n = Math.min(len, nChars - nextChar); //取实际读取字符数与目标字符数len的最小数        
        System.arraycopy(cb, nextChar, cbuf, off, n);  //从字符缓冲区中复制字符到目标数组中        
        nextChar += n; //字符缓冲区存储下标位置前诺,避免重复取一样数据        
        return n;
    } 
 

实际流程图解:图解read1(cbuf, off, len)方法即可,本质是该方法在起作用

4)读一行文字并返回该行字符,若读到文件末尾,则返回null:即当遇到换行符('\ n'),回车符('\ r')时会终止读取表示该行文字读取完毕且返回该行文字(不包含换行符和回车符)


   
    public String readLine() throws IOException
    {
        return readLine(false);
    }
 
    String readLine(boolean ignoreLF) throws IOException
    {
        StringBuffer s = null;
        int startChar;
 
        synchronized (lock)
        {
            ensureOpen();
            boolean omitLF = ignoreLF || skipLF;   
            bufferLoop: for (;;)
            {
 
                if (nextChar >= nChars) //判断是否有元素,没有则调用fill()方法取元素
                    fill();
                if (nextChar >= nChars)  //判断是否已到文件末尾,若到文件末尾,则返回S
                { 
                    if (s != null && s.length() > 0)
                        return s.toString();
                    else
                        return null;
                }
                boolean eol = false;
                char c = 0;
                int i;
 
               //如果遇到过换行符,则跳过该换行符继续读取
                if (omitLF && (cb[nextChar] == '\n'))
                    nextChar++;
                skipLF = false;
                omitLF = false;
 
                charLoop: for (i = nextChar; i < nChars; i++)
                {
                    c = cb[i];
                    if ((c == '\n') || (c == '\r'))
                    {
                        eol = true;
                        break charLoop;
                    }
                }
 
                startChar = nextChar;
                nextChar = i;
 
                if (eol)
                {
                    String str;
                    if (s == null)
                    {
                        str = new String(cb, startChar, i - startChar);
                    }
                    else
                    {
                        s.append(cb, startChar, i - startChar);
                        str = s.toString();
                    }
                    nextChar++;
                    if (c == '\r')
                    {
                        skipLF = true;
                    }
                    return str;
                }
 
                if (s == null)
                    s = new StringBuffer(defaultExpectedLineLength);
                s.append(cb, startChar, i - startChar);
            }
        }
    }

实际流程图解:

5)close()方法:关闭资源释放链接


    public void close() throws IOException
    {
        synchronized (lock)
        {
            if (in == null)
                return;
            in.close();
            in = null;
            cb = null;
        }
    }

6)其它的skip()、make()方法等暂时不了解

五、BufferedReader类与InputStreamReader类比较

InputStreamReader中的文档说明提到过:为了获得最高效率,请考虑在BufferedReader中包装InputStreamReader?

从read()方法理解,若使用InputStreamReader的read()方法,可以发现存在每2次就会调用一次解码器解码,但若是使用 BufferedReader包装InputStreamReader后调用read()方法,可以发现只会调用一次解码器解码,其余时候都是直接从BufferedReader的缓冲区中取字符即可

从read(char cbuf[], int offset, int length)方法理解,若使用InputStreamReader的方法则只会读取leng个字符,但是使用BufferedReader类则会读取读取8192个字符,会尽量提取比当前操作所需的更多字节;

例如文件中有20个字符,我们先通过read(cbuf,0,5)要读取5个字符到数组cbuf中,然后再通过read()方法读取1个字符。那么使用InputStreamReader类的话,则会调用一次解码器解码然后存储5个字符到数组中,然后又调用read()方法调用一次解码器读取2个字符,然后返回1个字符;等于是调用了2次解码器,若使用BufferedReader类的话则是先调用一次解码器读取20个字符到字符缓冲区中,然后复制5个到数组中,在调用read()方法时,则直接从缓冲区中读取字符,等于是调用了一次解码器

因此可以看出BufferedReader类会尽量提取比当前操作所需的更多字节,以应该更多情况下的效率提升,

因此在设计到文件字符输入流的时候,我们使用BufferedReader中包装InputStreamReader类即可

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

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Java基础知识之BufferedReader流的使用

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android 基础知识之 VelocityTracker使用

VelocityTracker是Android中用于追踪触摸事件速度的工具。它可以用于计算触摸事件的速度和加速度,从而用于实现一些手势操作,如滑动、拖拽等。使用VelocityTracker时,需要按照以下步骤进行操作:1. 创建Veloc
2023-09-14

Java基础知识中ByteArrayOutputStream流的使用方法是什么

Java基础知识中ByteArrayOutputStream流的使用方法是什么,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。Java ByteArrayOutputStream
2023-06-22

Java基础知识中StringWriter流的使用方法是什么

本篇文章为大家展示了Java基础知识中StringWriter流的使用方法是什么,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。Java StringWriter流的使用一、StringWriter流
2023-06-22

SAP SD基础知识之流程概览

SAP SD基础知识之流程概览 销售流程 销售流程从建立客户关系开始,至开出销售发票终止。售前活动Pre-Sales Activities(客户询、报价)--> 销售订单处理sales order processing(产生销售订单)--
2023-06-05

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录