Java中IO流分成兩大類,一種是輸入流,所有的輸入流都直接或間接繼承自InputStream抽象類,輸入流作為資料的來源,我們可以通過輸入流的read方法讀取位元組資料;
另一種是輸出流,所有的輸出流都直接或間接繼承自OutputStream抽象類,輸出流接收資料,可以通過write方法寫入位元組資料
在Java的IO流類中,大部分的輸入流和輸出流都是成對存在的,即如果存在XXXInputStream,那麼就存在XXXOutputStream,反之亦然。(SequenceInputStream和StringBufferInputStream是特例,沒有對應的SequenceOutputStream類和StringBufferOutputStream類)。許多IO操作都可能會丟擲IOException異常,比如read、write、close操作。
Java IO的一般使用原則:
一、按資料來源(去向)分類:
1、是檔案: FileInputStream, FileOutputStream, FileReader, FileWriter
2、是byte[]:ByteArrayInputStream, ByteArrayOutputStream
3、是Char[]: CharArrayReader, CharArrayWriter
4、是String: StringBufferInputStream, StringReader, StringWriter
5、網路資料流:InputStream, OutputStream, Reader, Writer
二、按是否格式化輸出分:
1、要格式化輸出:PrintStream, PrintWriter
三、按是否要緩衝分:
1、要緩衝:BufferedInputStream, BufferedOutputStream, BufferedReader, BufferedWriter
四、按資料格式分:
1、二進位制格式(只要不能確定是純文字的): InputStream, OutputStream及其所有帶Stream結束的子類
2、純文字格式(含純英文與漢字或其他編碼方式);Reader, Writer及其所有帶Reader, Writer的子類
五、按輸入輸出分:
1、輸入:Reader, InputStream型別的子類
2、輸出:Writer, OutputStream型別的子類
六、特殊需要:
1、從Stream到Reader,Writer的轉換類:InputStreamReader, OutputStreamWriter
2、物件輸入輸出:ObjectInputStream, ObjectOutputStream
3、程式間通訊:PipeInputStream, PipeOutputStream, PipeReader, PipeWriter
4、合併輸入:SequenceInputStream
5、更特殊的需要:PushbackInputStream, PushbackReader, LineNumberInputStream, LineNumberReader
決定使用哪個類以及它的構造程式的一般準則如下(不考慮特殊需要):
第一,考慮最原始的資料格式是什麼:是否為文字?
第二,是輸入還是輸出?
第三,是否需要轉換流:InputStreamReader, OutputStreamWriter?
第四,資料來源(去向)是什麼:檔案?記憶體?網路?
第五,是否要緩衝:bufferedReader (特別註明:一定要注意的是readLine()是否有定義,有什麼比read, write更特殊的輸入或輸出方法)
第六,是否要格式化輸出:print?
總結二: 首先是java的IO。
這破東西可真費事,I/O類庫常使用”流(stream)”這種抽象。所謂”流”是一種能生成或接受資料的,代表資料的源和目標的物件。流把I/O裝置內部的具體操作給隱藏起來了。 正如JDK文件所顯示的,Java的I/O類庫分成輸入和輸出兩大部分。所有InputStream和Reader的派生類都有一個基本的,繼承下來的,能讀取單個或byte陣列的read( )方法。同理,所有OutputStream和Writer的派生類都有一個基本的,能寫入單個或byte陣列的write( )方法。但通常情況下,你是不會去用這些方法的;它們是給其它類用的 —— 而後者會提供一些更實用的介面。因此,你很少會碰到只用一個類就能建立一個流的情形,實際上你得把多個物件疊起來,並以此來獲取所需的功能。Java的流類庫之所以會那麼讓人犯暈,最主要的原因就是”你必須為建立一個流而動用多個物件”。
Java的IO類結構: 根介面是InputStream/OutputStream,充當資料來源的IO類有FileInputStream/FileOutputStream,ByteArrayInputStream / ByteArrayOutputStream 等,充當裝飾功能的IO類有BufferedInputStream / BufferedOutputStream,DataInputStream / DataOutputStream等,
它們都是繼承裝飾介面FilterInputStream/FilterOutputStream。
使用IO時,首先建立一個資料來源IO,然後根據需要的功能建立裝飾類IO,其建構函式的引數為已建立的資料來源IO。
我們以建立一個具有緩衝的檔案輸入流為例,假定需要從磁碟讀取檔案“C:\log.txt”:
// 建立一個FileInputStream:
FileInputStream fileInput = new FileInputStream(”C:\\log.txt”);
// 建立一個BufferedInputStream:
BufferedInputStream bufferedInput = new BufferedInputStream(fileInput);
// 現在得到的bufferedInput即是具有緩衝的檔案輸入流,或者進一步簡寫如下:
InputStream input = new BufferedInputStream(new FileInputStream(”C:\\log.txt”)); // 現在得到的input即是具有緩衝的檔案輸入流
java.io.Reader 和 java.io.InputStream 區別
java.io.Reader 和 java.io.InputStream 組成了 Java 輸入類。Reader 用於讀入16位字元,也就是 Unicode 編碼的字元;而 InputStream 用於讀入 ASCII 字元和二進位制資料。
在 Java 中,有不同型別的 Reader 輸入流對應於不同的資料來源:
FileReader 用於從檔案輸入;
CharArrayReader 用於從程式中的字元陣列輸入;
StringReader 用於從程式中的字串輸入;
PipedReader 用於讀取從另一個執行緒中的 PipedWriter 寫入管道的資料。
相應的也有不同型別的 InputStream 輸入流對應於不同的資料來源:FileInputStream,ByteArrayInputStream,StringBufferInputStream,PipedInputStream。另外,還有兩種沒有對應 Reader 型別的 InputStream 輸入流:
Socket 用於套接字;
URLConnection 用於 URL 連線。
這兩個類使用 getInputStream() 來讀取資料。
相應的,java.io.Writer 和 java.io.OutputStream 也有類似的區別。
1、Java技術支援兩種資料型別的流
InputStream和OutputStream:位元組流。其它位元組流都是InputStream或OutputStream的子類。 Reader和 Writer:字元流。其它字元流都是Reader或Writer的子類。
2、節點流
Java 2 SDK中有三種基本型別的節點:檔案(file)、記憶體(memory)、管道(pipe)。
3、過程流
過程流在其它流之上,完成排序、變換等操作。過程流也被稱做過濾流。 當你需要改變輸入流的原始資料時,你可以將一個過濾輸入流連線到一個原始的輸入流上。 用過濾流將原始資料變換成你需要的格式。
4、基本位元組流類
4.1、FileInputStream和FileOutputStream
這兩個節點流用來操縱磁碟檔案。這些類的建構函式允許你指定它們所連線的檔案。
要構造一個FileInputStream,所關聯的檔案必須存在而且是可讀的。
如果你要構造一個FileOutputStream而輸出檔案已經存在,則它將被覆蓋。
FileInputStream infile = new FileInputStream(”myfile.dat”);
FileOutputStream outfile = new FileOutputStream(”results.dat”);
4.1、 BufferInputStream和BufferOutputStream
這些是過濾器流,它們可以提高I/O操作的效率。
4.3、 PipedInputStream和PipedOutputStream
管道流用來線上程間進行通訊。一個執行緒的PipedInputStream物件從另一個執行緒的PipedOutputStream物件讀取輸入。
要使管道流有用,必須有一個輸入方和一個輸出方。
4.4、 DataInputStream和DataOutputStream
這些過濾器通過流來讀寫Java基本類
5、 基本字元流類
圖闡述了Reader和Writer字元流的體系結構。
5.1、InputStreamReader 和 OutputStreamWriter
用於位元組流與字元流之間的轉換介面。
當你構造一個InputStreamReader或OutputStreamWriter時,轉換規則定義了16位Unicode和其它平臺的特定表示之間的轉換。
InputStreamReader從一個資料來源讀取位元組,並自動將其轉換成Unicode字元。
如果你特別宣告,InputStreamReade會將位元組流轉換成其它種類的字元流。 OutputStreamWriter將字元的Unicode編碼寫到輸出流,如果你的使用的不是Unicode字元,OutputStreamWriter會將你的字元編碼轉換成Unicode編碼。
5.2.、緩衝讀者和作者 因為在各種格式之間進行轉換和其它I/O操作很類似,所以在處理大塊資料時效率最高。
在InputStreamReader和OutputStreamWriter的結尾連結一個BufferedReader和BufferedWriter是一個好主意。
記住對BufferedWriter使用flush()方法。
5.3、 使用其它字元轉換
如果你需要從一個非本地(例如,從連線到一個不同型別的機器的網路連線讀取)的字元編碼讀取輸入, 你可以象下面這個程式那樣,使用顯式的字元編碼構造ir=new InputStreamReader(System.in, “8859_1″);
注:如果你通過網路連線讀取字元,就應該使用這種形式。
否則,你的程式會總是試圖將所讀取的字元當作本地表示來進行轉換,而這並不總是正確的。ISO 8859-1是對映到ASCII的Latin-1編碼模式。
6、 物件序列化 java.io.Serializable介面支援將一個Java技術物件存放到一個流中。
將一個物件存放到某種型別的永久儲存器上稱為”保持”。
如果一個物件可以被存放到磁碟或磁帶上,或者可以傳送到另外一臺機器並存放到儲存器或磁碟上,那麼這個物件就被稱為可保持的。
java.io.Serializable介面沒有任何方法,它只作為一個”標記”,用來表明實現了這個介面的類可以序列化。 類中沒有實現Serializable介面的物件不能被保持。
// 檔案實現追加:
// 其中的FileWriter()中的第二個引數的含義是:是否在檔案中追加內容
PrintWriter out = new PrintWriter(new FileWriter(logFileName, true), true);
Java讀寫檔案最常用的類是FileInputStream/FileOutputStream和FileReader/FileWriter。
其中FileInputStream和FileOutputStream是基於位元組流的,常用於讀寫二進位制檔案。
讀寫字元檔案建議使用基於字元的FileReader和FileWriter,省去了位元組與字元之間的轉換。
但這兩個類的建構函式預設使用系統的編碼方式,如果檔案內容與系統編碼方式不一致,可能會出現亂碼。
在這種情況下,建議使用FileReader和FileWriter的父類:
InputStreamReader/OutputStreamWriter, 它們也是基於字元的,但在建構函式中可以指定編碼型別:InputStreamReader(InputStream in, Charset cs) 和OutputStreamWriter(OutputStream out, Charset cs)。
// 讀寫檔案的編碼:
InputStreamReader r = new InputStreamReader(new FileInputStream(fileName), “utf-8″);
OutputStreamWriter out = new OutputStreamWriter(new FileOutputStream(fileName),”utf-8″);
/** 三種IO效能比較: 在讀寫一個10k檔案的時候,三種方式的耗時如下:
InputStreamReader And OutputStreamWriter : 63ms(可以設定檔案的編碼,如果不用buffer)
BufferedReader And BufferedWriter : 31ms
BufferedInputStream And BufferedOutputStream : 16ms