淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系
前言
這兩者在往文件中寫入字符串時(shí),最終都需要通過(guò)字符集的映射關(guān)系得到對(duì)應(yīng)字節(jié)。
但這二者在通過(guò)char得到對(duì)應(yīng)若干字節(jié)的時(shí)機(jī)不一樣,以new PrintStream( new BufferedOutputStream( new FileOutputStream("BasicFileOutput.out")));
和new PrintWriter( new BufferedWriter( new FileWriter("BasicFileOutput.out")))
為例,前者在存字符串時(shí),從PrintStream傳到BufferedOutputStream時(shí)就已經(jīng)是字節(jié)了;后者在存字符串時(shí),直到FileWriter真正寫入文件時(shí),才將字符轉(zhuǎn)換為字節(jié)。
如果PrintStream被設(shè)置為autoFlush,那么這些情況flush方法將會(huì)自動(dòng)執(zhí)行:寫入字節(jié)數(shù)組、任何重載版本的println被調(diào)用、一個(gè)換行符(char)被寫入、一個(gè)換行符的字節(jié)存儲(chǔ)(\n
)被寫入。
如果PrintWriter被設(shè)置為autoFlush,那么這些情況flush方法將會(huì)自動(dòng)執(zhí)行:println、printf、format方法被調(diào)用。
它們都不會(huì)拋出IO異常,因?yàn)樗鼈冊(cè)诜椒▋?nèi)部捕獲住了,可以通過(guò)checkError()來(lái)判斷是否發(fā)生異常。
PrintWriter會(huì)使用平臺(tái)特有的換行符(比如Windows和linux),PrintStream則固定使用\n
。
總的來(lái)說(shuō),Reader/Writer相比InputStream/OutputStream算是一種升級(jí),將當(dāng)初設(shè)計(jì)得不好的地方進(jìn)行了優(yōu)化。
二者的構(gòu)造器分析
這二者真的很像,你可以去看它們倆的api文檔,可以發(fā)現(xiàn)它們的構(gòu)造器和方法幾乎一模一樣。
你會(huì)注意到PrintWriter的構(gòu)造器中,也可以指定字符集,這可能有點(diǎn)奇怪,因?yàn)樽鳛橛脕?lái)裝飾的Writer(PrintWriter)來(lái)說(shuō),它應(yīng)該不需要關(guān)心字符應(yīng)該怎么對(duì)應(yīng)到字節(jié)上去,只需要關(guān)心字符就好了呀。
PrintWriter有兩個(gè)構(gòu)造器都可以指定字符集:PrintWriter(File file, String csn)
和PrintWriter(String fileName, String csn)
,但這兩個(gè)構(gòu)造器最終都會(huì)調(diào)用到下面這個(gè)構(gòu)造器,可以看出charset是給裝飾器最里層的FileOutputStream使用的,而外層的兩個(gè)Writer都不用關(guān)心。這說(shuō)明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來(lái)驅(qū)動(dòng)的(這看起來(lái)和上面貼的PrintWriter的構(gòu)造器里的邏輯一樣,這里指裝飾器的裝飾過(guò)程):
/* 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(); } } ... }
也就是說(shuō),這句new PrintStream( new BufferedOutputStream( new FileOutputStream("BasicFileOutput.out")));
實(shí)際上會(huì)產(chǎn)生五個(gè)流對(duì)象(裝飾器模式會(huì)套五層,禁止套娃?。???磜rite函數(shù),每次寫入字符串時(shí),竟然是先通過(guò)裝飾器最外層的BufferedWriter進(jìn)行的寫入,然后再執(zhí)行BufferedWriter、OutputStreamWriter的flushBuffer函數(shù)把字符轉(zhuǎn)換為字節(jié)(像擠牙膏一樣,只不過(guò)這是從外往里擠),并且將字節(jié)弄到this對(duì)象里面。
PrintStream的三種類型(File、OutputStream、String)的構(gòu)造器都可以帶字符集。
我們隨便看一個(gè)帶字符集的構(gòu)造器,發(fā)現(xiàn)字符集被設(shè)置在this的外面一層流上,所以通過(guò)OutputStreamWriter向this傳遞字節(jié)數(shù)組時(shí),這個(gè)字節(jié)數(shù)組就已經(jīng)經(jīng)過(guò)了特定字符集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),那就是它還可以接受一個(gè)Writer。
二者的方法分析
對(duì)比二者的api文檔,你會(huì)發(fā)現(xiàn)前面的方法簽名,這二者都是一模一樣,除了后面的write方法們,所以我們只看不同的部分。相同的部分占大部分,同學(xué)們可以自行查看。
PrintStream的write方法們
從方法的描述也可以看出,這些write方法都是針對(duì)byte或者byte[ ]的。
發(fā)現(xiàn)有一個(gè)write方法是從FilterOutputStream繼承而來(lái),因?yàn)檫@個(gè)重載版本的write方法PrintStream沒(méi)有去重寫父類方法(下面這兩個(gè)就是重寫了FilterOutputStream了的)。
FilterOutputStream的write(byte[] b)
版本如下:
PrintWriter的write方法們
從方法的描述也可以看出,這些write方法都是針對(duì)char或者string的。
到此這篇關(guān)于淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系的文章就介紹到這了,更多相關(guān)PrintStream和PrintWriter區(qū)別內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java利用線程工廠監(jiān)控線程池的實(shí)現(xiàn)示例
這篇文章主要介紹了Java利用線程工廠監(jiān)控線程池的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04Java運(yùn)用設(shè)計(jì)模式中的建造者模式構(gòu)建項(xiàng)目的實(shí)例解析
這篇文章主要介紹了Java運(yùn)用設(shè)計(jì)模式中的建造者模式構(gòu)建項(xiàng)目的實(shí)例解析,建造者模式對(duì)外隱藏創(chuàng)建過(guò)程的產(chǎn)品,使用組合的方式,由指揮者來(lái)決定建造的流程,需要的朋友可以參考下2016-04-04springboot2.1.7整合thymeleaf代碼實(shí)例
這篇文章主要介紹了springboot2.1.7整合thymeleaf代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12Spring boot監(jiān)控Actuator-Admin實(shí)現(xiàn)過(guò)程詳解
這篇文章主要介紹了Spring boot監(jiān)控Actuator-Admin實(shí)現(xiàn)過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-09-09SpringBoot+layuimini實(shí)現(xiàn)左側(cè)菜單動(dòng)態(tài)展示的示例代碼
Layuimini是Layui的升級(jí)版,它是專業(yè)做后臺(tái)頁(yè)面的框架,而且是適合PC端和移動(dòng)端,以下地址可以在PC端顯示,也可以在手機(jī)上顯示,只不過(guò)會(huì)做自適應(yīng),本文將給大家介紹了SpringBoot+layuimini實(shí)現(xiàn)左側(cè)菜單動(dòng)態(tài)展示的方法,需要的朋友可以參考下2024-04-04MybatisPlus實(shí)現(xiàn)邏輯刪除功能
這篇文章主要介紹了MybatisPlus實(shí)現(xiàn)邏輯刪除功能,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12SpringBoot實(shí)現(xiàn)前后端、json數(shù)據(jù)交互以及Controller接收參數(shù)的幾種常用方式
這篇文章主要給大家介紹了關(guān)于SpringBoot實(shí)現(xiàn)前后端、json數(shù)據(jù)交互以及Controller接收參數(shù)的幾種常用方式,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-03-03