溫馨提示×

溫馨提示×

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

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

淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系

發(fā)布時間:2020-10-10 14:34:59 來源:腳本之家 閱讀:195 作者:anlian523 欄目:編程語言

前言

這兩者在往文件中寫入字符串時,最終都需要通過字符集的映射關(guān)系得到對應(yīng)字節(jié)。

但這二者在通過char得到對應(yīng)若干字節(jié)的時機(jī)不一樣,以new PrintStream( new BufferedOutputStream( new FileOutputStream("BasicFileOutput.out")));new PrintWriter( new BufferedWriter( new FileWriter("BasicFileOutput.out")))為例,前者在存字符串時,從PrintStream傳到BufferedOutputStream時就已經(jīng)是字節(jié)了;后者在存字符串時,直到FileWriter真正寫入文件時,才將字符轉(zhuǎn)換為字節(jié)。

如果PrintStream被設(shè)置為autoFlush,那么這些情況flush方法將會自動執(zhí)行:寫入字節(jié)數(shù)組、任何重載版本的println被調(diào)用、一個換行符(char)被寫入、一個換行符的字節(jié)存儲(\n)被寫入。

如果PrintWriter被設(shè)置為autoFlush,那么這些情況flush方法將會自動執(zhí)行:println、printf、format方法被調(diào)用。

它們都不會拋出IO異常,因?yàn)樗鼈冊诜椒▋?nèi)部捕獲住了,可以通過checkError()來判斷是否發(fā)生異常。

PrintWriter會使用平臺特有的換行符(比如Windows和linux),PrintStream則固定使用\n。

總的來說,Reader/Writer相比InputStream/OutputStream算是一種升級,將當(dāng)初設(shè)計(jì)得不好的地方進(jìn)行了優(yōu)化。

二者的構(gòu)造器分析

這二者真的很像,你可以去看它們倆的api文檔,可以發(fā)現(xiàn)它們的構(gòu)造器和方法幾乎一模一樣。

你會注意到PrintWriter的構(gòu)造器中,也可以指定字符集,這可能有點(diǎn)奇怪,因?yàn)樽鳛橛脕硌b飾的Writer(PrintWriter)來說,它應(yīng)該不需要關(guān)心字符應(yīng)該怎么對應(yīng)到字節(jié)上去,只需要關(guān)心字符就好了呀。

PrintWriter有兩個構(gòu)造器都可以指定字符集:PrintWriter(File file, String csn)PrintWriter(String fileName, String csn),但這兩個構(gòu)造器最終都會調(diào)用到下面這個構(gòu)造器,可以看出charset是給裝飾器最里層的FileOutputStream使用的,而外層的兩個Writer都不用關(guān)心。這說明Writer確實(shí)不需要關(guān)心字符集。

 /* Private constructor */
  private PrintWriter(Charset charset, File file)
    throws FileNotFoundException
  {
    this(new BufferedWriter(new OutputStreamWriter(new FileOutputStream(file), charset)),
       false);
  }

其實(shí)PrintStream也很low,觀察它的構(gòu)造器可以發(fā)現(xiàn)它竟然是靠BufferedWriter來驅(qū)動的(這看起來和上面貼的PrintWriter的構(gòu)造器里的邏輯一樣,這里指裝飾器的裝飾過程):

/* Private constructors */
  private PrintStream(boolean autoFlush, OutputStream out) {
    super(out);
    this.autoFlush = autoFlush;
    this.charOut = new OutputStreamWriter(this);//往自己身上裝飾一層
    this.textOut = new BufferedWriter(charOut);//往自己身上再裝飾一層
  }

  private void write(String s) {
    try {
      synchronized (this) {
        ensureOpen();
        textOut.write(s);//先調(diào)用最外層的流的write函數(shù)
        textOut.flushBuffer();
        charOut.flushBuffer();
        if (autoFlush && (s.indexOf('\n') >= 0))
          out.flush();
      }
    }
  ...
  }

也就是說,這句new PrintStream( new BufferedOutputStream( new FileOutputStream("BasicFileOutput.out")));實(shí)際上會產(chǎn)生五個流對象(裝飾器模式會套五層,禁止套娃?。?磜rite函數(shù),每次寫入字符串時,竟然是先通過裝飾器最外層的BufferedWriter進(jìn)行的寫入,然后再執(zhí)行BufferedWriter、OutputStreamWriter的flushBuffer函數(shù)把字符轉(zhuǎn)換為字節(jié)(像擠牙膏一樣,只不過這是從外往里擠),并且將字節(jié)弄到this對象里面。

PrintStream的三種類型(File、OutputStream、String)的構(gòu)造器都可以帶字符集。

淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系

我們隨便看一個帶字符集的構(gòu)造器,發(fā)現(xiàn)字符集被設(shè)置在this的外面一層流上,所以通過OutputStreamWriter向this傳遞字節(jié)數(shù)組時,這個字節(jié)數(shù)組就已經(jīng)經(jīng)過了特定字符集charset的encode了:

 private PrintStream(boolean autoFlush, OutputStream out, Charset charset) {
    super(out);
    this.autoFlush = autoFlush;
    this.charOut = new OutputStreamWriter(this, charset);
    this.textOut = new BufferedWriter(charOut);
  }

PrintWriter相比PrintStream多了一種類型的構(gòu)造器(File、OutputStream、String、Writer),那就是它還可以接受一個Writer。

淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系

二者的方法分析

對比二者的api文檔,你會發(fā)現(xiàn)前面的方法簽名,這二者都是一模一樣,除了后面的write方法們,所以我們只看不同的部分。相同的部分占大部分,同學(xué)們可以自行查看。

PrintStream的write方法們

從方法的描述也可以看出,這些write方法都是針對byte或者byte[ ]的。

發(fā)現(xiàn)有一個write方法是從FilterOutputStream繼承而來,因?yàn)檫@個重載版本的write方法PrintStream沒有去重寫父類方法(下面這兩個就是重寫了FilterOutputStream了的)。

淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系

FilterOutputStream的write(byte[] b)版本如下:

淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系

PrintWriter的write方法們

從方法的描述也可以看出,這些write方法都是針對char或者string的。

淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系

到此這篇關(guān)于淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系的文章就介紹到這了,更多相關(guān)PrintStream和PrintWriter區(qū)別內(nèi)容請搜索億速云以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持億速云!

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

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

AI