一、BufferedReader類
. 所屬類庫:
java.lang.Object
java.io.Reader
java.io.BufferedReader
. 基本概念 :
public class BufferedReader extends Reader
從字符輸入流中讀取文本,緩沖各個(gè)字符,從而實(shí)現(xiàn)字符、數(shù)組和行的高效讀取。 可以指定緩沖區(qū)的大小,或者可使用默認(rèn)的大小。大多數(shù)情況下,默認(rèn)值足夠大。
通常, Reader 所作的每個(gè)讀取請(qǐng)求都會(huì)導(dǎo)致對(duì)底層字符或字節(jié)流進(jìn)行相應(yīng)的讀取請(qǐng)求。因此,建議用 BufferedReader 包裝所有其 read() 操作可能開銷很高的 Reader (如 FileReader 和 InputStreamReader )。
BufferedReader 流能夠讀取文本行 , 通過向 BufferedReader 傳遞一個(gè) Reader 對(duì)象 , 來創(chuàng)建一個(gè) BufferedReader 對(duì)象 , 之所以這樣做是因?yàn)?FileReader 沒有提供讀取文本行的功能 .
. Demo :
通過 Bufferedreader 捕獲所輸入的語句 :
1
2
3
4
5
6
7
8
9
10
|
import java.io.*; class BufferedReaderDemo{ public static void main(String[] args) throws IOException { BufferedReader bufferedReader = new BufferedReader( new InputStreamReader(System.in)); System.out.print( "請(qǐng)輸入一系列文字,可包括空格:" ); String text =bufferedReader.readLine(); System.out.println( "請(qǐng)輸入文字:" +text); } } |
注解:
throws IOException 拋出異常
InputStreamReader 是字節(jié)流通向字符流的橋梁
二、InputStreamReader類
InputStreamReader 將字節(jié)流轉(zhuǎn)換為字符流。是字節(jié)流通向字符流的橋梁。如果不指定字符集編碼,該解碼過程將使用平臺(tái)默認(rèn)的字符編碼,如:GBK。
構(gòu)造方法 :
InputStreamReader isr = new InputStreamReader(InputStream in);//構(gòu)造一個(gè)默認(rèn)編碼集的InputStreamReader類
InputStreamReader isr = new InputStreamReader(InputStream in,String charsetName);//構(gòu)造一個(gè)指定編碼集的
InputStreamReader類。
參數(shù) in對(duì)象通過 InputStream in = System.in;獲得。//讀取鍵盤上的數(shù)據(jù)。
或者 InputStream in = new FileInputStream(String fileName);//讀取文件中的數(shù)據(jù)。可以看出 FileInputStream 為InputStream的子類。
主要方法 :int read();//讀取單個(gè)字符。
int read(char []cbuf);//將讀取到的字符存到數(shù)組中。返回讀取的字符數(shù)。
. Demo :
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
|
import java.io.*; class InputStreamReaderDemo { public static void transReadNoBuf() throws IOException { /** * 沒有緩沖區(qū),只能使用read()方法。 */ //讀取字節(jié)流 //InputStream in = System.in;//讀取鍵盤的輸入。 InputStream in = new FileInputStream( "D:\\demo.txt" ); //讀取文件的數(shù)據(jù)。 //將字節(jié)流向字符流的轉(zhuǎn)換。要啟用從字節(jié)到字符的有效轉(zhuǎn)換, //可以提前從底層流讀取更多的字節(jié). InputStreamReader isr = new InputStreamReader(in); //讀取 //綜合到一句。 //InputStreamReader isr = new InputStreamReader( //new FileInputStream("D:\\demo.txt")); char []cha = new char [ 1024 ]; int len = isr.read(cha); System.out.println( new String(cha, 0 ,len)); isr.close(); } public static void transReadByBuf() throws IOException { /** * 使用緩沖區(qū) 可以使用緩沖區(qū)對(duì)象的 read() 和 readLine()方法。 */ //讀取字節(jié)流 //InputStream in = System.in;//讀取鍵盤上的數(shù)據(jù) InputStream in = new FileInputStream( "D:\\demo.txt" ); //讀取文件上的數(shù)據(jù)。 //將字節(jié)流向字符流的轉(zhuǎn)換。 InputStreamReader isr = new InputStreamReader(in); //讀取 //創(chuàng)建字符流緩沖區(qū) BufferedReader bufr = new BufferedReader(isr); //緩沖 //BufferedReader bufr = new BufferedReader( //new InputStreamReader(new FileInputStream("D:\\demo.txt")));可以綜合到一句。 /*int ch =0; ch = bufr.read(); System.out.println((char)ch); */ String line; while ((line = bufr.readLine())!= null ){ System.out.println(line); } isr.close(); } } |
三、InputStreamReader、BufferedReader真實(shí)案例(非編碼集)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
import java.io.*; class UtilResource { private void initializeResource() { try { //讀取文件,并且以u(píng)tf-8的形式寫出去 BufferedReader bufread; String read; bufread = new BufferedReader( new InputStreamReader(ResourceHelper .getResourceInputStream( "pinyin.txt" ))); while ((read = bufread.readLine()) != null ) { System.out.println(read); } bufread.close(); } catch (FileNotFoundException ex) { ex.printStackTrace(); } catch (IOException ex) { ex.printStackTrace(); } } } |
注:其中 pinyin.txt 放于項(xiàng)目的根目錄下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
import java.io.BufferedInputStream; import java.io.FileInputStream; import java.io.FileNotFoundException; class ResourceHelper { /** * @param resourceName * @return * @return */ static BufferedInputStream getResourceInputStream(String resourceName) { try { return new BufferedInputStream( new FileInputStream(resourceName)); } catch (FileNotFoundException e) { // TODO Auto-generated catch block e.printStackTrace(); } return null ; } } |
總結(jié):
InputStreamReader 類
是字節(jié)流通向字符流的橋梁,封裝了InputStream在里頭, 它以較高級(jí)的方式,一次讀取一個(gè)一個(gè)字符,以文本格式輸入 / 輸出,可以指定編碼格式;
BufferedReader 類
BufferedReader 由Reader類擴(kuò)展而來,提供通用的緩沖方式文本讀取,而且提供了很實(shí)用的readLine,讀取一個(gè)文本行,從字符輸入流中讀取文本,緩沖各個(gè)字符,從而提供字符、數(shù)組和行的高效讀取。
ps:InputStream、InputStreamReader和Reader的關(guān)系
InputStream:得到的是字節(jié)輸入流,InputStream.read("filename")之后,得到字節(jié)流
Reader:讀取的是字符流
InputStreamReader:從字節(jié)到字符的橋梁。InputStreamReader(InputStream.read("filename"));
reader.read(InputStreamReader(InputStream in));便可從字節(jié)變?yōu)樽址蛴★@示了。
java.io.Reader 和 java.io.InputStream 組成了Java 輸入類。
Reader 用于讀入16位字符,也就是Unicode 編碼的字符;而 InputStream 用于讀入 ASCII 字符和二進(jìn)制數(shù)據(jù)。
Reader支持16位的Unicode字符輸出,InputStream支持8位的字符輸出。
Reader和InputStream分別是I/O庫提供的兩套平行獨(dú)立的等級(jí)機(jī)構(gòu),1byte = 8bits InputStream、OutputStream是用來處理8位元的流,Reader、Writer是用來處理16位元的流。
而在JAVA語言中,byte類型是8位的,char類型是16位的,所以在處理中文的時(shí)候需要用Reader和Writer。
值得說明的是,在這兩種等級(jí)機(jī)構(gòu)下,還有一道橋梁InputStreamReader、OutputStreamWriter負(fù)責(zé)進(jìn)行InputStream到Reader的適配和由OutputStream到Writer的適配。
在 Java中,有不同類型的 Reader 輸入流對(duì)應(yīng)于不同的數(shù)據(jù)源:
FileReader 用于從文件輸入; CharArrayReader 用于從程序中的字符數(shù)組輸入; StringReader 用于從程序中的字符串輸入; PipedReader 用于讀取從另一個(gè)線程中的 PipedWriter 寫入管道的數(shù)據(jù)。
相應(yīng)的也有不同類型的 InputStream 輸入流對(duì)應(yīng)于不同的數(shù)據(jù)源:FileInputStream,ByteArrayInputStream,StringBufferInputStream,PipedInputStream。
另外,還有兩種沒有對(duì)應(yīng) Reader 類型的 InputStream 輸入流: Socket 用于套接字; URLConnection 用于 URL 連接。 這兩個(gè)類使用 getInputStream() 來讀取數(shù)據(jù)。
相應(yīng)的,java.io.Writer 和 java.io.OutputStream 也有類似的區(qū)別。
關(guān)于InputStream.read(byte[] b)和InputStream.read(byte[] b,int off,int len)這兩個(gè)方法都是用來從流里讀取多個(gè)字節(jié)的,有經(jīng)驗(yàn)的程序員就會(huì)發(fā)現(xiàn),這兩個(gè)方法經(jīng)常 讀取不到自己想要讀取的個(gè)數(shù)的字節(jié)。比如第一個(gè)方法,程序員往往希望程序能讀取到b.length個(gè)字節(jié),而實(shí)際情況是,系統(tǒng)往往讀取不了這么多。仔細(xì)閱讀Java的API說明就發(fā)現(xiàn)了,這個(gè)方法 并不保證能讀取這么多個(gè)字節(jié),它只能保證最多讀取這么多個(gè)字節(jié)(最少1個(gè))。因此,如果要讓程序讀取count個(gè)字節(jié),最好用以下代碼:
1
2
3
4
5
|
byte [] b = new byte [count]; int readCount = 0 ; // 已經(jīng)成功讀取的字節(jié)的個(gè)數(shù) while (readCount < count) { readCount += in.read(bytes, readCount, count - readCount); } |
用這段代碼可以保證讀取count個(gè)字節(jié),除非中途遇到IO異常或者到了數(shù)據(jù)流的結(jié)尾(EOFException)