一.input和output
1.stream代表的是任何有能力產出數據的數據源,或是任何有能力接收數據的接收源。
在java的io中,所有的stream(包括input和out stream)都包括兩種類型:
1.1 以字節為導向的stream
以字節為導向的stream,表示以字節為單位從stream中讀取或往stream中寫入信息。以字節為導向的stream包括下面幾種類型:
1) input stream:
1) bytearrayinputstream:把內存中的一個緩沖區作為inputstream使用
2) stringbufferinputstream:把一個string對象作為inputstream
3) fileinputstream:把一個文件作為inputstream,實現對文件的讀取操作
4) pipedinputstream:實現了pipe的概念,主要在線程中使用
5) sequenceinputstream:把多個inputstream合并為一個inputstream
2) out stream
1) bytearrayoutputstream:把信息存入內存中的一個緩沖區中
2) fileoutputstream:把信息存入文件中
3) pipedoutputstream:實現了pipe的概念,主要在線程中使用
4) sequenceoutputstream:把多個outstream合并為一個outstream
1.2 以unicode字符為導向的stream
以unicode字符為導向的stream,表示以unicode字符為單位從stream中讀取或往stream中寫入信息。以unicode字符為導向的stream包括下面幾種類型:
1) input stream
1) chararrayreader:與bytearrayinputstream對應
2) stringreader:與stringbufferinputstream對應
3) filereader:與fileinputstream對應[來源 www.iocblog.net]
4) pipedreader:與pipedinputstream對應
2) out stream
1) chararraywrite:與bytearrayoutputstream對應
2) stringwrite:無與之對應的以字節為導向的stream
3) filewrite:與fileoutputstream對應
4) pipedwrite:與pipedoutputstream對應
以字符為導向的stream基本上對有與之相對應的以字節為導向的stream。兩個對應類實現的功能相同,字是在操作時的導向不同。如chararrayreader:和bytearrayinputstream的作用都是把內存中的一個緩沖區作為inputstream使用,所不同的是前者每次從內存中讀取一個字節的信息,而后者每次從內存中讀取一個字符。
1.3 兩種不現導向的stream之間的轉換
inputstreamreader和outputstreamreader:把一個以字節為導向的stream轉換成一個以字符為導向的stream。
2. stream添加屬性
2.1 “為stream添加屬性”的作用
運用上面介紹的java中操作io的api,我們就可完成我們想完成的任何操作了。但通過filterinputstream和filteroutstream的子類,我們可以為stream添加屬性。下面以一個
例子來說明這種功能的作用。
如果我們要往一個文件中寫入數據,我們可以這樣操作:
fileoutstream fs = new fileoutstream(“test.txt”);
然后就可以通過產生的fs對象調用write()函數來往test.txt文件中寫入數據了。但是,如果我們想實現“先把要寫入文件的數據先緩存到內存中,再把緩存中的數據寫入文件中”的功能時,上面的api就沒有一個能滿足我們的需求了。但是通過filterinputstream和filteroutstream的子類,為fileoutstream添加我們所需要的功能。
2.2 filterinputstream的各種類型
2.2.1 用于封裝以字節為導向的inputstream
1) datainputstream:從stream中讀取基本類型(int、char等)數據。
2) bufferedinputstream:使用緩沖區
3) linenumberinputstream:會記錄input stream內的行數,然后可以調用getlinenumber()和setlinenumber(int)
4) pushbackinputstream:很少用到,一般用于編譯器開發
2.2.2 用于封裝以字符為導向的inputstream
1) 沒有與datainputstream對應的類。除非在要使用readline()時改用bufferedreader,否則使用datainputstream
2) bufferedreader:與bufferedinputstream對應
3) linenumberreader:與linenumberinputstream對應
4) pushbackreader:與pushbackinputstream對應
2.3 filteroutstream的各種類型
2.2.3 用于封裝以字節為導向的outputstream
1) dataioutstream:往stream中輸出基本類型(int、char等)數據。
2) bufferedoutstream:使用緩沖區
3) printstream:產生格式化輸出
2.2.4 用于封裝以字符為導向的outputstream
1) bufferedwrite:與對應
2) printwrite:與對應
3. randomaccessfile
1) 可通過randomaccessfile對象完成對文件的讀寫操作
2) 在產生一個對象時,可指明要打開的文件的性質:r,只讀;w,只寫;rw可讀寫
3) 可以直接跳到文件中指定的位置
4. i/o應用的一個例子
import java.io.*;
public class testio{
public static void main(string[] args)
throws ioexception{
//1.以行為單位從一個文件讀取數據
bufferedreader in =
new bufferedreader(
new filereader("f:\nepalon\testio.java"));
string s, s2 = new string();
while((s = in.readline()) != null)
s2 += s + " ";
in.close();
//1b. 接收鍵盤的輸入
bufferedreader stdin =
new bufferedreader(
new inputstreamreader(system.in));
system.out.println("enter a line:");
system.out.println(stdin.readline());
//2. 從一個string對象中讀取數據
stringreader in2 = new stringreader(s2);
int c;
while((c = in2.read()) != -1)
system.out.println((char)c);
in2.close();
//3. 從內存取出格式化輸入
try{
DataInputStream in3 =
new DataInputStream(
new ByteArrayInputStream(s2.getBytes()));
while(true)
System.out.println((char)in3.readByte());
}
catch(EOFException e){
System.out.println("End of stream");
}
//4. 輸出到文件
try{
BufferedReader in4 = new BufferedReader(new StringReader(s2));
PrintWriter out1 =
new PrintWriter(
new BufferedWriter(
new FileWriter("F:\\nepalon\\ TestIO.out")));
int lineCount = 1;
while((s = in4.readLine()) != null)
out1.println(lineCount++ + ":" + s);
out1.close();
in4.close();
}
catch(EOFException ex){
System.out.println("End of stream");
}
//5. 數據的存儲和恢復
try{
DataOutputStream out2 =
new DataOutputStream(
new BufferedOutputStream(
new FileOutputStream("F:\\nepalon\\ Data.txt")));
out2.writeDouble(3.1415926);
out2.writeChars("\nThas was pi:writeChars\n");
out2.writeBytes("Thas was pi:writeByte\n");
out2.close();
DataInputStream in5 =
new DataInputStream(
new BufferedInputStream(
new FileInputStream("F:\\nepalon\\ Data.txt")));
BufferedReader in5br =
new BufferedReader(
new InputStreamReader(in5));
System.out.println(in5.readDouble());
System.out.println(in5br.readLine());
System.out.println(in5br.readLine());
}
catch(EOFException e){
System.out.println("End of stream");
}
//6. 通過RandomAccessFile操作文件
RandomAccessFile rf =
new RandomAccessFile("F:\\nepalon\\ rtest.dat", "rw");
for(int i=0; i<10; i++)
rf.writeDouble(i*1.414);
rf.close();[來源 www.iocblog.net]
rf = new RandomAccessFile("F:\\nepalon\\ rtest.dat", "r");
for(int i=0; i<10; i++)
System.out.println("Value " + i + ":" + rf.readDouble());
rf.close();
rf = new RandomAccessFile("F:\\nepalon\\ rtest.dat", "rw");
rf.seek(5*8);
rf.writeDouble(47.0001);
rf.close();
rf = new RandomAccessFile("F:\\nepalon\\ rtest.dat", "r");
for(int i=0; i<10; i++)
System.out.println("Value " + i + ":" + rf.readDouble());
rf.close();
}
}
關于代碼的解釋(以區為單位):
1區中,當讀取文件時,先把文件內容讀到緩存中,當調用in.readLine()時,再從緩存中以字符的方式讀取數據(以下簡稱“緩存字節讀取方式”)。
1b區中,由于想以緩存字節讀取方式從標準IO(鍵盤)中讀取數據,所以要先把標準IO(System.in)轉換成字符導向的stream,再進行BufferedReader封裝。
2區中,要以字符的形式從一個String對象中讀取數據,所以要產生一個StringReader類型的stream。
4區中,對String對象s2讀取數據時,先把對象中的數據存入緩存中,再從緩沖中進行讀??;對TestIO.out文件進行操作時,先把格式化后的信息輸出到緩存中,再把緩存中的信息輸出到文件中。
5 區中,對Data.txt文件進行輸出時,是先把基本類型的數據輸出屋緩存中,再把緩存中的數據輸出到文件中;對文件進行讀取操作時,先把文件中的數據讀取到緩存中,再從緩存中以基本類型的形式進行讀取。注意in5.readDouble()這一行。因為寫入第一個writeDouble(),所以為了正確顯示。也要以基本類型的形式進行讀取。
6區是通過RandomAccessFile類對文件進行操作。