2007年1月8日


File,FileInputStream,FileReader,InputStreamReader,BufferedReader 的使用和區(qū)別

參考資料:

l???????? 《 core java 》 12 章

l???????? 使用 Java 操作文本文件的方法詳解

http://java.ccidnet.com/art/3737/20041108/523627_1.html

l? FileReader 是什么類?和 FileInputStream 有什么不同?????????

?http://book.hackbase.com/ask2/ask107572.htm
自己的整理和領(lǐng)會(huì):
引言:
????
C語言只需要一個(gè)File*就可以了,與C不同,java有一系列流類型,其數(shù)量超過60種。類庫的設(shè)計(jì)者聲稱:“有

足夠的理由為用戶提供豐富的流類型的選擇:這樣做可以減少程序的錯(cuò)誤?!崩纾贑語言種,許多人認(rèn)為“

將輸出流寫入一個(gè)只讀模式的文件”是很常見的錯(cuò)誤。(事實(shí)上,這并不常見。)

我們認(rèn)為在C++語言中,流接口設(shè)計(jì)者避免程序出錯(cuò)的主要“工具”是小心謹(jǐn)慎的態(tài)度,在java語言中更是如

此。流庫的高度復(fù)雜性迫使程序設(shè)計(jì)人員謹(jǐn)小慎微。
????
1.? File類
??
1 ) File 類介紹(《 core java 》 638 頁)

File 類封裝了對(duì)用戶機(jī)器的文件系統(tǒng)進(jìn)行操作的功能。例如,可以用 File 類獲得文件上次修改的時(shí)間移動(dòng),

或者對(duì)文件進(jìn)行刪除、重命名。換句話說,流類關(guān)注的是文件內(nèi)容,而 File 類關(guān)注的是文件在磁盤上的存儲(chǔ)

File 類的主要方法有: getName(),getCanonicalFile(),lastModified(),isDerector(),isFile(),getPath()

等;

2 ) File 類與 FileInputStream 類的區(qū)別:

流類關(guān)注的是文件內(nèi)容,而 File 類關(guān)注的是文件在磁盤上的存儲(chǔ)。

File 不屬于文件流 , 只能代表一個(gè)文件或是目錄的路徑名而已。

提示:(《 core java 》 639 頁)

如果處理文件或者目錄名,就應(yīng)該使用 File 對(duì)象,而不是字符串。例如, File 類的 equals 方法知道一些

文件系統(tǒng)對(duì)大小寫是敏感的,目錄尾的“ / ”字符無關(guān)緊要。

自己的領(lǐng)會(huì):

FileInputStream 類或者 FileReader 類的構(gòu)造函數(shù)有多個(gè),其中典型的兩個(gè)分別為:一個(gè)使用 File 對(duì)象為

參數(shù);而另一個(gè)使用表示路徑的 String 對(duì)象作為參數(shù);自己以前一直覺得直接用了 String 指定路徑就可以

了,一直不明白為什么很多人都先構(gòu)造一個(gè) File 對(duì)象,現(xiàn)在終于明白了,“如果處理文件或者目錄名,就應(yīng)

該使用 File 對(duì)象,而不是字符串?!保?

2.?????? FileInputStream 類

1 ) FileInputStream 類介紹:

以字節(jié)為單位(非 unicode )的流處理。字節(jié)序列即:二進(jìn)制數(shù)據(jù)。與編碼無關(guān),不存在亂碼問題。

FileInputStream 類的主要方法有:

Read (), read ( byte[] b ), read ( byte[],int off,int len ) ,available();

2 ) FileInputStream 類與 FileReader 類的區(qū)別:

兩個(gè)類的構(gòu)造函數(shù)的形式和參數(shù)都是相同的,參數(shù)為 File 對(duì)象或者表示路徑的 String ,它們到底有何區(qū)別

呢?

l???????? Readers and Writers work only on line based character data, so plain text files.
For anything else, you MUST use Streams.

l???????? JDK5 API:

FileInputStream is meant for reading streams of raw bytes such as image data. For reading streams

of characters, consider using FileReader.

FileReader is meant for reading streams of characters. For reading streams of raw bytes, consider

using a FileInputStream .

l???????? FileInputStream :以字節(jié)流方式讀取; FileReader :把文件轉(zhuǎn)換為字符流讀入;
l??????? InputStream提供的是字節(jié)流的讀取,而非文本讀取,這是和Reader類的根本區(qū)別。用Reader讀取出

來的是char數(shù)組或者String ,使用InputStream讀取出來的是byte數(shù)組。
l??????? Reader類及其子類提供的字符流的讀取char(16位,unicode編碼),inputStream及其子類提供字節(jié)

流的讀取byte(8位),所以FileReader類是將文件按字符流的方式讀取,F(xiàn)ileInputStream則按字節(jié)流的方式

讀取文件;InputStreamReader可以將讀如stream轉(zhuǎn)換成字符流方式,是reader和stream之間的橋梁
l? 最初Java是不支持對(duì)文本文件的處理的,為了彌補(bǔ)這個(gè)缺憾而引入了Reader和Writer兩個(gè)類。
??
l???????? FileInputStream 類以二進(jìn)制輸入 / 輸出, I/O 速度快且效率搞,但是它的 read ()方法讀到

的是一個(gè)字節(jié)(二進(jìn)制數(shù)據(jù)),很不利于人們閱讀。

l???????? 而 FileReader 類彌補(bǔ)了這個(gè)缺陷,可以以文本格式輸入 / 輸出,非常方便;比如可以使用

while((ch = filereader.read())!=-1 ) 循環(huán)來讀取文件;可以使用 BufferedReader 的 readLine() 方法一

行一行的讀取文本。

l???????? 當(dāng)我們讀寫文本文件的時(shí)候,采用 Reader 是非常方便的,比如 FileReader ,

InputStreamReader 和 BufferedReader 。其中最重要的類是 InputStreamReader ,它是字節(jié)轉(zhuǎn)換為字符的橋

梁。 你可以在構(gòu)造器重指定編碼的方式,如果不指定的話將采用底層操作系統(tǒng)的默認(rèn)編碼方式,例如 GBK 等

。

l???????? FileReader 與 InputStreamReader 涉及編碼轉(zhuǎn)換 ( 指定編碼方式或者采用 os 默認(rèn)編碼 ) ,可

能在不同的平臺(tái)上出現(xiàn)亂碼現(xiàn)象!而 FileInputStream 以二進(jìn)制方式處理,不會(huì)出現(xiàn)亂碼現(xiàn)象 .

3 )自己的領(lǐng)會(huì):

l???????? 如果處理純文本文件,建議使用 FileReader ,因?yàn)楦奖?,也更適合閱讀;但是要注意編碼問題

!
l?? 其他情況(處理非純文本文件),F(xiàn)ileInputStream是唯一的選擇;FileInputStream是進(jìn)Socket通訊時(shí)會(huì)

用到很多,如將文件流是Stream的方式傳向服務(wù)器!
??
3.?????? FileReader 類

1)??? FileReader 類介紹:

InputStreamReader 類的子類,所有方法( read ()等)都從父類 InputStreamReader 中繼承而來;

2)??? 與 InputStreamReader 類的區(qū)別:

l???????? 自己的領(lǐng)會(huì):

該類與它的父類 InputStreamReader 的主要不同在于構(gòu)造函數(shù),主要區(qū)別也就在于構(gòu)造函數(shù)!從

InputStreamReader 的構(gòu)造函數(shù)中看到,參數(shù)為 InputStream 和編碼方式,可以看出,當(dāng)要指定編碼方式時(shí),

必須使用 InputStreamReader 類;而 FileReader 構(gòu)造函數(shù)的參數(shù)與 FileInputStream 同,為 File 對(duì)象或

表示 path 的 String ,可以看出,當(dāng)要根據(jù) File 對(duì)象或者 String 讀取一個(gè)文件時(shí),用 FileReader ;我

想 FileReader 子類的作用也就在于這個(gè)小分工吧。

3)??? 一般用法:

FileReader fr = new FileReader("ming.txt");
   char[] buffer = new char[1024];
   int ch = 0;
   while((ch = fr.read())!=-1 )
   {
    System.out.print((char)ch);
   }

4.?????? InputStreamReader 類

l???????? 以文本格式輸入 / 輸出,可以指定編碼格式;

l???????? 主要方法:

getEncoding (),read();

l???????? 一般用法:

InputStreamReader isr = new InputStreamReader(new FileInputStream("ming.txt"));
   while((ch = isr.read())!=-1)
   {
    System.out.print((char)ch);
   }

5.?????? BufferedReader 類

l???????? Jdk5 api :

Read text from a character-input stream, buffering characters so as to provide for the efficient

reading of characters, arrays, and lines.
l??? BufferedReader 由Reader類擴(kuò)展而來,提供通用的緩沖方式文本讀取,而且提供了很實(shí)用的readLine,

讀取分行文本很適合,BufferedReader是針對(duì)Reader的,不直接針對(duì)文件,也不是只針對(duì)文件讀取。
l? 一般用法:
????
BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream("ming.txt")));
  String data = null;
  while((data = br.readLine())!=null)
  {
   System.out.println(data);
  }
????
??
6.?????? 總結(jié)以上內(nèi)容,得出比較好的規(guī)范用法:

1)??? File file = new File ("hello.txt");

FileInputStream in=new FileInputStream(file);

2)??? File file = new File ("hello.txt");

FileInputStream in=new FileInputStream(file);

InputStreamReader inReader=new InputStreamReader(in);

BufferedReader bufReader=new BufferedReader(inReader);

3)??? File file = new File ("hello.txt");

FileReader fileReader=new FileReader(file);

BufferedReader bufReader=new BufferedReader(fileReader);

7.一些寫法的區(qū)別:
1)
File file = new File ("hello.txt");
FileInputStream in=new FileInputStream(file);
InputStreamReader inReader=new InputStreamReader(in);
BufferedReader bufReader=new BufferedReader(inReader);

2)
FileInputStream in=null;
File file = new File ("hello.txt");
in=new FileInputStream(file);
BufferedReader bufReader=new BufferedReader(new InputStreamReader(in));

3)
File file = new File ("hello.txt");
BufferedReader bufReader=new BufferedReader(new InputStreamReader(new FileInputStream(file)));

上述兩種寫法的微小區(qū)別:
a)第二種方式中把“FileInputStream in=null;”定義單獨(dú)放在開始處,說明下面應(yīng)該還有要用到in對(duì)象變量的地方;(BufferedReader處用了)

b)第二種方式?jīng)]有定義InputStreamReader的對(duì)象變量,直接在BufferedReader的構(gòu)造函數(shù)中new一個(gè),
這種方式與第一種方式的主要區(qū)別:InputStreamReader對(duì)象只使用一次!

這對(duì)于在這里只需要使用一次這個(gè)InputStreamReader對(duì)象的應(yīng)用來說更好;無需定義InputStreamReader的對(duì)象變量,接收由new返回的該對(duì)象的引用,因?yàn)橄旅娴某绦蛑胁恍枰@個(gè)InputStreamReader的對(duì)象變量,所以無需定義;所以這種情況下,第二種方式比第一種更好一些。

c)第三種方式中,典型的三層嵌套委派關(guān)系,清晰看出Reader的委派模式(《corejava》12章有圖描述該委派關(guān)系),F(xiàn)ileInputStream和InputStreamReader都沒有定義變量,new生成的對(duì)象都只是使用一次。

d)三種方式的區(qū)別也就在于FileInputStream和InputStreamReader對(duì)象是否都只使用一次,是否需要定義它們的對(duì)象變量,以及個(gè)人的編碼習(xí)慣。

e)但是要注意異常處理,F(xiàn)ileInputStream(file)會(huì)拋出NotFileFoundException,如果采用surround方式
(try&catch)處理,應(yīng)該用第二種方式,這樣可以用System.out.println提示文件未找到;
當(dāng)然在函數(shù)名后使用throws Exception,然后用第三種方式也行,但似乎這適合有用戶界面的情況,把異常拋出在客戶端在處理。