偷個懶,直接參考上篇字符輸入流Reader的形式,其實Reader和Writer本來就大同小異;
本篇將對JAVA I/O流中的字符輸出流Writer做個簡單的概括:
總得來說,每個字符輸出流類都有一個對應的用途,如下:
再看下Writer的組成結構(下圖,包括字符輸入流Reader);
下面將對各個字符輸出流做個詳細介紹;
OutputStreamWriter繼承字符流抽象類Writer,每個構造方法都包含一個字節流OutputStream入參,可以讓我們往字節輸出流中寫入字符;
PS:無論是構造方法裡對字節流的處理(StreamDecoder.forOutputStreamWriter),還是字符流的讀取(StreamDecoder.write),底層都是通過StreamDecoder類實現的,有興趣的可以深入了解~
舉個例子,如下,為方便閱讀,不做異常處理:
package com.pichen.io; import java.io.IOException; import java.io.OutputStreamWriter; import java.io.Writer; public class IOTest { public static void main(String[] args) throws IOException { //通過“標准”字節輸出流得到一個字符輸出流 Writer writer = new OutputStreamWriter(System.out); //往字符流寫入字符 writer.write("hello~"); //關閉流 writer.close(); } }
FileWriter繼承字符流OutputStreamWriter,往本地文件寫入字符,查看其構造方法,都是先根據傳入的參數生成一個FileOutputStream字節流對象,然後調用父類OutputStreamWriter的構造方法得到字符流;
舉個簡單的例子:
package com.pichen.io; import java.io.FileWriter; import java.io.IOException; import java.io.Writer; public class IOTest { public static void main(String[] args) throws IOException { //得到字符輸出流 Writer writer = new FileWriter("C:\\test.txt"); //寫入字符 writer.write('a'); //刷新該流的緩沖 writer.flush(); //關閉流 writer.close(); } }
CharArrayReader可以讓我們往字符數組中寫入字符,在構造方法裡可以指定字符數組大小,且後續在write的過程中,會自動增大該字符數組;
舉個簡單的例子:
package com.pichen.io; import java.io.CharArrayWriter; import java.io.IOException; public class IOTest { public static void main(String[] args) throws IOException { //創建字符流,指定初始緩沖區大小 CharArrayWriter writer = new CharArrayWriter(6); //寫入字符 writer.write("aaabbb"); //從字符輸出流中得到字符數組 System.out.println(writer.toCharArray()); //關閉流 writer.close(); } }
可以通過PipedWriter和PipedReader創建字符流管道,線程間可以通過管道進行通信,注意:一定要同一個JVM中的兩個線程;
PipedWriter一般是要和PipedReader配合使用的,其中一個線程通過PipedWriter往管道寫數據,另一個線程通過PipedReader從管道讀數據,注意讀寫都會阻塞線程,如下示例:
package com.pichen.io; import java.io.IOException; import java.io.PipedReader; import java.io.PipedWriter; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class IOTest { public static void main(String[] args) throws IOException { final PipedWriter pw = new PipedWriter(); final PipedReader pr = new PipedReader(pw); ExecutorService es = Executors.newFixedThreadPool(2); es.execute(new Runnable() { @Override public void run() { try { pw.write("hello~"); } catch (IOException e) { e.printStackTrace(); } } }); es.execute(new Runnable() { @Override public void run() { char[] cbuffer = new char[6]; try { // 會導致線程阻塞 pr.read(cbuffer, 0, 6); } catch (IOException e) { e.printStackTrace(); } System.out.println(cbuffer); } }); } }
這個字符流可以用來裝飾其它字符輸出流,可以為其它字符輸出流提供字符輸出緩沖區,到時一次性寫入,避免每次都一個字節一個字節往外部媒介中寫入數據,這裡用到了設計模式裡的裝飾器模式,可以參考我之前寫的,
http://www.cnblogs.com/chenpi/p/5173818.html
被裝飾的字符流可以有更多的行為,比如newLine方法等;
舉個使用的例子,讀取外部文件:
package com.pichen.io; import java.io.BufferedWriter; import java.io.FileWriter; import java.io.IOException; public class IOTest { public static void main(String[] args) throws IOException { // 創建一個緩沖字符輸出流,這裡設定緩沖區大小為10k BufferedWriter writer = new BufferedWriter(new FileWriter("C:\\test.txt"), 10 * 1024); // 往緩沖區寫入字符 writer.write("aaa"); writer.newLine(); writer.write("bbb"); // 關閉流 writer.close(); } }
抽象類FilterWriter是實現自定義過濾輸出字符流的基類,從源碼的實現上來看,僅僅只是簡單覆蓋了Writer中的所有方法,感覺沒什麼卵用,因為已經有一個抽象類Writer了;
字符輸出流,可以構造字符串,其內部是通過StringBuffer實現的,
舉個簡單的使用例子
package com.pichen.io; import java.io.IOException; import java.io.StringWriter; public class IOTest { public static void main(String[] args) throws IOException { StringWriter writer = new StringWriter(); writer.write("hello "); writer.write("world~"); System.out.println(writer.toString()); } }
允許將格式化數據寫入到字符輸出流中,如下示例,將格式化數據寫到控制台上;
package com.pichen.io; import java.io.IOException; import java.io.PrintWriter; public class IOTest { public static void main(String[] args) throws IOException { PrintWriter writer = new PrintWriter(System.out); writer.write("hello "); writer.write("world~"); writer.printf("%5d", 101); writer.close(); } }