前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >算法与编程之 -- IO流

算法与编程之 -- IO流

作者头像
Java深度编程
发布2020-06-10 15:55:17
2990
发布2020-06-10 15:55:17
举报
文章被收录于专栏:Java深度编程

编写一个程序,将 a.txt 文件中的单词与 b.txt 文件中的单词交替合并到 c.txt文件中,a.txt 文件中的单词用回车符分隔,b.txt 文件中用回车或空格进行分隔。

如果你在面试中,遇到这个题了,千万不要慌,因为出题者的初衷可能并不是要你立马写出这个程序,而是想考虑你的思维能力。事实上短时间内也很难写出这个代码。

首先我们要想到解题思路:1.先用字符流读取到a.txt和b.txt文件,同时创建一个输出流用于输出到c.txt;2.其次你需要先循环输出a到c,但每输出一个a文本要间隔的输出一个b到c,而你又怎么知道什么时候的b对应于那个a呢?那么我们就需要建立一个标识,用于标记a和b都输出到哪里了。

废话不多说,直接上代码:

代码语言:javascript
复制
package com.wenxue;

import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
public class Main {

    public static void main(String[] args) throws Exception{
        FileManager a = new FileManager("C:\\a.txt");
        FileManager b = new FileManager("C:\\b.txt");
        FileWriter c = new FileWriter("C:\\c.txt");
        String aWord = null;
        String bWord = null;
        while((aWord = a.nextWord()) !=null ){
            c.write(aWord + "\n");
            bWord = b.nextWord();
            if(bWord != null){
                c.write(bWord+ "\n ");
            }
        }
        //这里之所以接着循环,是为了防止a已经输出完了,但是b还没有输出完
        while((bWord= b.nextWord()) != null){
            c.write(bWord+ "\n ");
        }
        c.close();
    }

    static class FileManager {
        String[] arr  = null;
        //定义一个标记用于记录a,b输出的位置
        int i = 0;

        public FileManager(String filename) throws Exception {
            File f = new File(filename);
            FileReader reader = new FileReader(f);
            //将文件内容转换成字符数组
            char[] buf = new char[(int) f.length()];
            int len = reader.read(buf);
            //还可以进一步将文件内容转成字符串
            //String results = new String(buf, 0, len);
            arr = new String[len];
            for (int j = 0; j < buf.length; j++) {
                arr[j] = String.valueOf(buf[j]);
            }
        }

        public String nextWord() {
            if (i == arr.length){
                return null;
            }
            return arr[i++];
        }
    }
}

从上面我们可以看到,先将文件内容转成了字符char数组,再讲它转成了字符串String数组,然后根据标记 i 从而实现了间隔输出的功能,同时为了防止b的内容大过a的内容,因此在下方继续循环输出b到c。有人可能觉得为啥要讲数组转来转去,其实也是可以不转的,但编写上就有些别扭,如下:

代码语言:javascript
复制
package com.wenxue;

import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
public class Main {

    public static void main(String[] args) throws Exception{
        FileManager2 a= new FileManager2("C:\\a.txt");
        FileManager2 b= new FileManager2("C:\\b.txt");
        FileWriter c= new FileWriter("C:\\c.txt");
        char aWord ;
        char bWord ;
        while((aWord = a.nextWord()) != ' ' ){
            c.write(aWord + "\n");
            bWord = b.nextWord();
            if(bWord != ' '){
                c.write(bWord+ "\n ");
            }
        }
        //这里之所以接着循环,是为了防止a已经输出完了,但是b还没有输出完
        while((bWord= b.nextWord()) != ' '){
            c.write(bWord+ "\n ");
        }
        c.close();
    }
    
    static class FileManager2 {
        char[] chars ;
        int i = 0;

        public FileManager2(String filename) throws Exception {
            File f = new File(filename);
            FileReader reader = new FileReader(f);
            //将文件内容转换成字符数组
            char[] buf = new char[(int) f.length()];
            int len = reader.read(buf);
            chars = buf;
        }

        public char nextWord() {
            if (i == chars.length){
                return ' ';
            }
            return chars[i++];
        }
    }
}

从上我们可以看到不适用String数组,适用char数组也是能够实现了,只是写法上要判断是否等于空字符‘ ’,给人的感觉有些别扭而已。

易错点:将文件流转成字符数组的时候,一定不能少的一行代码:

代码语言:javascript
复制
//这样代码不能少
int len = reader.read(buf);

为什么这行代码不能少?debug你会发现没走这行代码的时候的字符数组虽然有值,但里面的内容却都是空的,必须要走完reader.read(buf)后才可以有值,那又是为什么呢?翻阅底层代码,你会看到如下代码:

代码语言:javascript
复制
public int read(char[] var1, int var2, int var3) throws IOException {
    int var4 = var2;
    int var5 = var3;
    Object var6 = this.lock;
    synchronized(this.lock) {
        this.ensureOpen();
        if (var4 >= 0 && var4 <= var1.length && var5 >= 0 && var4 + var5 <= var1.length && var4 + var5 >= 0) {
            if (var5 == 0) {
                return 0;
            } else {
                byte var7 = 0;
                if (this.haveLeftoverChar) {
                    var1[var4] = this.leftoverChar;
                    ++var4;
                    --var5;
                    this.haveLeftoverChar = false;
                    var7 = 1;
                    if (var5 == 0 || !this.implReady()) {
                        return var7;
                    }
                }

                if (var5 == 1) {
                    int var8 = this.read0();
                    if (var8 == -1) {
                        return var7 == 0 ? -1 : var7;
                    } else {
                        var1[var4] = (char)var8;
                        return var7 + 1;
                    }
                } else {
                    return var7 + this.implRead(var1, var4, var4 + var5);
                }
            }
        } else {
            throw new IndexOutOfBoundsException();
        }
    }
 }

我们可以看到是在底层代码里进行了赋值,所以必须执行哪一行代码会,字符数组里才真正的有值了。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-11-30,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java深度编程 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档