流 JAVA /IO 基本小結
經過一行常見的代碼討論:new BufferedReader(new InputStreamReader(System.in))
java的IO是基於流(stream)概念的,什麼是流呢,做爲初學者,
我是這樣理解的,在各個應用之間傳送的是BITS,這些BIT可已被認爲是流體,能夠就認爲是水流,那麼用來在各個水源之間轉移水的工具應該選擇什麼呢?通常狀況下,水管是能夠的,因此數據我將數據源比做水源,將流對象比做水管
這樣就有了對流的第一步認識,它不再神祕了。
對於流,咱們要研究些什麼呢,咱們主要是針對應用掌握流的特性,而以後根據流的特性,咱們轉移不一樣的數據時,選擇不透的流對象,達到咱們的目的。
下面讓咱們從一行常見的代碼來分析流吧!
new BufferedReader(new InputStreamReader(System.in)),這是用來從鍵盤接受一行輸入的代碼,下面咱們從裏到外進行分析吧。
System.in的類型是InputStream,它表明的是鍵盤接受的輸入,就是說鍵盤是數據源;System.in的類型能夠歸結爲節點流、字節流、輸入流;接下來是InputStreamReader這個對象是處理流,字符流,輸入流;
最外面的是BufferedReader的類型是緩衝處理流、字符流、輸入流。是否是有點繞啊,下面咱們就從流的分類開始。
流的分類
(重點的經過分類記住這些流的模樣)
按照方向分類:
輸入流和輸出流
流的輸入輸出都是以應用程序爲基準的,這一點必定要注意。
輸入流,模樣很好記,通常狀況下,輸入流是帶有Input字樣或Reader字樣的,如FileInputStream和BufferedReader等等,這些都是輸入流。
輸出流,通常狀況下,是帶有Output字樣或Writer的,如FileOutputStream和FileWriter等等,詳細請查查API文檔,看看是否是這樣。
至於何時使用輸入流,何時使用輸出流,我想咱們就沒必要探討了吧!
按照處理的單位:
字節流和字符流
字節流,通常是帶有Stream字樣的,如InputStream,FileInputStream等等,這組流處理的最小單位之字節。
字符流,通常是帶有Reader或Writer字樣的,如InputStreamReader等等,它們處理的最小單位是字符。
按照數據的來源:
節點流和處理流
節點流的數據來源是應用程序、文件、鍵盤、等等,是非流對象來源,而處理流的數據來源是其餘流對象。
流的使用
一.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對應
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 + "/n";
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();
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類對文件進行操做。java