溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java?IO及BufferedReader.readline()出現(xiàn)的Bug該怎么處理

發(fā)布時(shí)間:2021-12-09 15:04:57 來源:億速云 閱讀:609 作者:柒染 欄目:開發(fā)技術(shù)

本篇文章為大家展示了Java IO及BufferedReader.readline()出現(xiàn)的Bug該怎么處理,內(nèi)容簡(jiǎn)明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲。

Java IO及BufferedReader.readline()的Bug

IO流

Java?IO及BufferedReader.readline()出現(xiàn)的Bug該怎么處理

:流是一組有序的,有起點(diǎn)和終點(diǎn)的字節(jié)集合,是對(duì)計(jì)算機(jī)中數(shù)據(jù)傳輸?shù)目偡Q。即數(shù)據(jù)在兩個(gè)設(shè)備間的傳輸稱為流,流的本質(zhì)是數(shù)據(jù)傳輸。

BufferedReader.readline()方法Bug

錯(cuò)誤代碼:

		File testTxt = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/test.txt");
		File file_copy3 = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/test_copy3.txt");
        BufferedReader bufferedReader = new BufferedReader(new FileReader(testTxt));
        BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter(file_copy3));
        //readLine() 每次讀取一行
        while (bufferedReader.readLine() != null){
            System.out.println(bufferedReader.readLine());
            bufferedWriter.write(bufferedReader.readLine());
        }
        bufferedWriter.close();
        bufferedReader.close();

原文件:

Java?IO及BufferedReader.readline()出現(xiàn)的Bug該怎么處理

結(jié)果:

Java?IO及BufferedReader.readline()出現(xiàn)的Bug該怎么處理

結(jié)果控制臺(tái)只打印了第二行,最后還報(bào)錯(cuò)了空指針異常

原因:

是代碼中每次調(diào)用readline()方法,就會(huì)向下讀取一行所以錯(cuò)誤代碼中表示的是while 判斷 的第一行不為null,打印的是第二行 ,然后寫入的是第三行,在次while判斷的是第四行 有內(nèi)容,打印的是第五行 為null,寫入的是第六行也為null,就導(dǎo)致了空指針異常。

修改后的代碼:

	 	String line;
        while ((line = bufferedReader.readLine()) != null){
            System.out.println(line);
            bufferedWriter.write(line);
        }

結(jié)果:

Java?IO及BufferedReader.readline()出現(xiàn)的Bug該怎么處理

注意:只用readline()復(fù)制后的但是和原文件是不同,沒有了換行符,如果需要可以在while循環(huán)體內(nèi)加上/r/n

Java?IO及BufferedReader.readline()出現(xiàn)的Bug該怎么處理

源碼

package com.lsh.io;
import java.io.*;
import java.time.Duration;
import java.time.Instant;
/**
 * @author :LiuShihao
 * @date :Created in 2021/3/3 11:09 上午
 * @desc :  只要是處理純文本數(shù)據(jù),就優(yōu)先考慮使用字符流。除此之外都使用字節(jié)流。
 *   
 */
public class FileIO {
    public static File testTxt = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/test.txt");
    public static File catImg = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/cat.jpg");
    public static void main(String[] args) throws Exception {
        Instant now = Instant.now();
        System.out.println("開始復(fù)制:"+now);
//        copyFile();
//        copyByReaderAndWriter1();
        copyByReaderAndWriter2();
        Instant end = Instant.now();
        // Duration  期間Instant          Period  時(shí)期  LocalDateTime
        System.out.println("復(fù)制完成:"+end+",耗時(shí):"+ Duration.between(now,end));
    }
    /**
     * 使用FileinputStream、FileOutputStream   與原文件一樣
     * 將一個(gè)文件復(fù)制一份
     */
    public static void  copyFile() throws Exception {
//        File file = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/test.txt");
        File file_copy1 = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/test_copy.txt");
//        File file_copy1 = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/cat_copy.jpg");
        FileInputStream fis = new FileInputStream(testTxt);
        FileOutputStream fos = new FileOutputStream(file_copy1);
        //byte[] bytes = new byte[1024];
        byte[] bytes = new byte[fis.available()];
        int read = fis.read(bytes);
        if (read != -1){
            fos.write(bytes);
        }
        fis.close();
        fos.close();
    }
    /**
     * 字符流
     * 使用FileInputStream、FileOutputStream、InputStreamReader、OutputStreamWriter、BufferedReader、BufferedWriter 復(fù)制文件
     *
     * readLine() 不用while的判斷只會(huì)輸出一行。
     */
    public static void copyByReaderAndWriter1() throws Exception {
//        File testTxt = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/test.txt");
        File file_copy2 = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/test_copy2.txt");
        FileInputStream fis = new FileInputStream(testTxt);
        FileOutputStream fos = new FileOutputStream(file_copy2);
        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(fis));
        BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(fos));
        String line;
        while ((line = bufferedReader.readLine()) != null){
            System.out.println(line);
            bufferedWriter.write(line);
        }
        // 注意: fis、fos要在bufferedWriter、bufferedReader關(guān)閉之后,否則會(huì)報(bào)錯(cuò)!
        bufferedWriter.close();
        bufferedReader.close();
        fis.close();
        fos.close();
    }
    /**
     * 字符流
     * 使用BufferedReader、BufferedWriter、FileReader、FileWriter復(fù)制文件
     * @throws Exception
     */
    public static void copyByReaderAndWriter2() throws Exception{
        File file_copy3 = new File("/Users/LiuShihao/IdeaProjects/netty_demo/src/main/resources/test_copy3.txt");
        BufferedReader bufferedReader = new BufferedReader(new FileReader(testTxt));
        BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter(file_copy3));
        //readLine() 每次讀取一行
        String line;
        while ((line = bufferedReader.readLine()) != null){
            System.out.println(line);
            bufferedWriter.write(line);
        }
        bufferedWriter.close();
        bufferedReader.close();
    }
}

使用BufferReader類的readLine()方法注意問題

一、BufferReader類的readLine()方法

public String readLine():直到程序遇到了換行符或者是對(duì)應(yīng)流的結(jié)束符,該方法才會(huì)認(rèn)為讀到了一行,才會(huì)結(jié)束其阻塞,讓程序繼續(xù)往下執(zhí)行。

注意:讀取到?jīng)]有數(shù)據(jù)時(shí)就返回null(因?yàn)槠渌黵ead()方法當(dāng)讀到?jīng)]有數(shù)據(jù)時(shí)返回-1),而實(shí)際上readLine()是一個(gè)阻塞函數(shù),當(dāng)沒有數(shù)據(jù)讀取時(shí),就一直會(huì)阻塞在那,而不是返回null。

讀取一個(gè)文本行,通過下列字符之一即可認(rèn)為某行已終止:換行 ('\n')、回車 ('\r') 或回車后直接跟著換行。

返回:到達(dá)流末尾,就返回null。

注意:當(dāng)循環(huán)讀取文件內(nèi)容時(shí),循環(huán)條件的結(jié)束要注意使用正確。

錯(cuò)誤的使用方式:

String valueString = null;
   while (bf.readLine()!=null){        //這樣會(huì)造成數(shù)據(jù)丟失,因?yàn)樵谶@里已經(jīng)調(diào)用了readLine()方法,已經(jīng)讀取了一行,下次調(diào)用時(shí),就會(huì)丟失一行。
        System.out.println(valueString);
}

正確的解決方法:用一個(gè)變量來接收方法的返回值

String valueString = null;
   while ((valueString=bf.readLine())!=null){     //通過變量來接收數(shù)據(jù),避免數(shù)據(jù)丟失
        System.out.println(valueString);
}

二、DataInputStream類的readUTF()方法

readUTF讀取的必須是writeUTF()寫下的字符串。即DataOutputStream的 writeUTF(String str)方法配套使用

上述內(nèi)容就是Java IO及BufferedReader.readline()出現(xiàn)的Bug該怎么處理,你們學(xué)到知識(shí)或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識(shí)儲(chǔ)備,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI