io包支持Java的基本I/O(輸入/輸出)系統(tǒng),包括文件的輸入/輸出。對(duì)輸入/輸出的支持是來(lái)源于Java的內(nèi)核API庫(kù),而不是語(yǔ)言關(guān)鍵字。
一、輸入/輸出基礎(chǔ)
很多實(shí)際的Java應(yīng)用程序不是基于文本的控制臺(tái)程序。盡管基于文本的程序作為教學(xué)實(shí)例是很出色的,它們無(wú)法勝任JAVA在實(shí)際中的重要應(yīng)用。Java對(duì)外設(shè)輸入/輸出的支持也是有限的,并且用起來(lái)有些笨拙——甚至是在簡(jiǎn)單的例子程序中?;谖谋镜目刂婆_(tái)輸入/輸出對(duì)于Java程序并不是十分重要。
Java 提供了與文件和網(wǎng)絡(luò)相關(guān)的強(qiáng)大的和靈活的輸入/輸出支持,Java的輸入/輸出系統(tǒng)是緊密相連并且是具有一致性的。
1.1 流的概念
Java程序通過(guò)流來(lái)完成輸入/輸出。流是生產(chǎn)或消費(fèi)信息的抽象。流通過(guò)Java的輸入/輸出系統(tǒng)與物理設(shè)備鏈接。盡管與它們鏈接的物理設(shè)備不盡相同,所有流的行為具有同樣的方式。這樣,相同的輸入/輸出類(lèi)和方法適用于所有類(lèi)型的外部設(shè)備。這意味著一個(gè)輸入流能夠抽象多種不同類(lèi)型的輸入:從磁盤(pán)文件,從鍵盤(pán)或從網(wǎng)絡(luò)套接字。同樣,一個(gè)輸出流可以輸出到控制臺(tái),磁盤(pán)文件或相連的網(wǎng)絡(luò)。流是處理輸入/輸出的一個(gè)潔凈的方法,例如它不需要代碼理解鍵盤(pán)和網(wǎng)絡(luò)的不同。Java中流的實(shí)現(xiàn)是在java.io包定義的類(lèi)層次結(jié)構(gòu)內(nèi)部的。
1.2 字節(jié)流和字符流
要使用流類(lèi),必須導(dǎo)入Java.io包。Java 2 定義了兩種類(lèi)型的流:字節(jié)類(lèi)和字符類(lèi)。字節(jié)流(byte stream)為處理字節(jié)的輸入和輸出提供了方便的方法。例如使用字節(jié)流讀取或書(shū)寫(xiě)二進(jìn)制數(shù)據(jù)。字符流(character stream)為字符的輸入和輸出處理提供了方便。它們采用了統(tǒng)一的編碼標(biāo)準(zhǔn),因而可以國(guó)際化。在某些場(chǎng)合,字符流比字節(jié)流更有效。在最底層,所有的輸入/輸出都是字節(jié)形式的?;谧址牧髦粸樘幚碜址峁┓奖阌行У姆椒?。下面是對(duì)字節(jié)流和字符流的概述。
1.2.1 字節(jié)流類(lèi)
字節(jié)流由兩個(gè)類(lèi)層次結(jié)構(gòu)定義。在頂層有兩個(gè)抽象類(lèi):InputStream 和 OutputStream。每個(gè)抽象類(lèi)都有多個(gè)具體的子類(lèi),這些子類(lèi)對(duì)不同的外設(shè)進(jìn)行處理,例如磁盤(pán)文件,網(wǎng)絡(luò)連接,甚至是內(nèi)存緩沖區(qū)。字節(jié)流類(lèi)顯示于表1-1中。
表1-1 字節(jié)流類(lèi)
流類(lèi) 含義
BufferedInputStream緩沖輸入流
BufferedOutputStream緩沖輸出流
ByteArrayInputStream從字節(jié)數(shù)組讀取的輸入流
ByteArrayOutputStream向字節(jié)數(shù)組寫(xiě)入的輸出流
DataInputStream包含讀取Java標(biāo)準(zhǔn)數(shù)據(jù)類(lèi)型方法的輸入流
DataOutputStream包含編寫(xiě)Java標(biāo)準(zhǔn)數(shù)據(jù)類(lèi)型方法的輸出流
FileInputStream讀取文件的輸入流
FileOutputStream寫(xiě)文件的輸出流
FilterInputStream實(shí)現(xiàn)InputStream
FilterOutputStream實(shí)現(xiàn)OutputStream
InputStream描述流輸入的抽象類(lèi)
OutputStream描述流輸出的抽象類(lèi)
PipedInputStream輸入管道
PipedOutputStream輸出管道
PrintStream包含print()和println()的輸出流
PushbackInputStream 支持向輸入流返回一個(gè)字節(jié)的單字節(jié)的“unget”的輸入流
RandomAccessFile支持隨機(jī)文件輸入/輸出
SequenceInputStream兩個(gè)或兩個(gè)以上順序讀取的輸入流組成的輸入流
抽象類(lèi)InputStream 和 OutputStream定義了實(shí)現(xiàn)其他流類(lèi)的關(guān)鍵方法。最重要的兩種方法是read()和write(),它們分別對(duì)數(shù)據(jù)的字節(jié)進(jìn)行讀寫(xiě)。兩種方法都在InputStream 和OutputStream中被定義為抽象方法。它們被派生的流類(lèi)重載。
1.2.2 字符流類(lèi)
字符流類(lèi)由兩個(gè)類(lèi)層次結(jié)構(gòu)定義。頂層有兩個(gè)抽象類(lèi):Reader和Writer。這些抽象類(lèi)處理統(tǒng)一編碼的字符流。Java中這些類(lèi)含有多個(gè)具體的子類(lèi)。字符流類(lèi)如表1-2所示。
表1-2 字符流的輸入/輸出類(lèi)
抽象類(lèi)Reader和Writer定義了幾個(gè)實(shí)現(xiàn)其他流類(lèi)的關(guān)鍵方法。其中兩個(gè)最重要的是read()和write(),它們分別進(jìn)行字符數(shù)據(jù)的讀和寫(xiě)。這些方法被派生流類(lèi)重載。
1.3 預(yù)定義流
所有的Java程序自動(dòng)導(dǎo)入java.lang包。該包定義了一個(gè)名為System的類(lèi),該類(lèi)封裝了運(yùn)行時(shí)環(huán)境的多個(gè)方面。System 同時(shí)包含三個(gè)預(yù)定義的流變量,in,out和err。這些成員在System中是被定義成public 和static型的,這意味著它們可以不引用特定的System對(duì)象而被用于程序的其他部分。
System.out是標(biāo)準(zhǔn)的輸出流。默認(rèn)情況下,它是一個(gè)控制臺(tái)。System.in是標(biāo)準(zhǔn)輸入,默認(rèn)情況下,它指的是鍵盤(pán)。System.err指的是標(biāo)準(zhǔn)錯(cuò)誤流,它默認(rèn)是控制臺(tái)。然而,這些流可以重定向到任何兼容的輸入/輸出設(shè)備。System.in 是inputStream的對(duì)象;System.out和System.err是PrintStream的對(duì)象。它們都是字節(jié)流,盡管它們用來(lái)讀寫(xiě)外設(shè)的字符。但可以用基于字符的流來(lái)包裝它們。
二、讀取控制臺(tái)輸入
在Java 1.0中,完成控制臺(tái)輸入的惟一途徑是字節(jié)流,使用該方法的老代碼依然存在。今天,運(yùn)用字節(jié)流讀取控制臺(tái)輸入在技術(shù)上仍是可行的,但這樣做需要用到不被贊成的方法,這種做法不值得推薦。Java 2中讀取控制臺(tái)輸入的首選方法是字符流,它使程序容易符合國(guó)際標(biāo)準(zhǔn)并且易于維護(hù)。
Java沒(méi)有像標(biāo)準(zhǔn)C的函數(shù)scanf()或C++輸入操作符那樣的統(tǒng)一的控制臺(tái)輸入方法。Java中,控制臺(tái)輸入由從System.in讀取數(shù)據(jù)來(lái)完成。為獲得屬于控制臺(tái)的字符流,在BufferedReader對(duì)象中包裝System.in。BufferedReader 支持緩沖輸入流。它最常見(jiàn)的構(gòu)造函數(shù)如下:
BufferedReader(Reader inputReader)
這里,inputReader是鏈接被創(chuàng)建的BufferedReader實(shí)例的流。Reader是一個(gè)抽象類(lèi)。它的一個(gè)具體的子類(lèi)是InputStreamReader,該子類(lèi)把字節(jié)轉(zhuǎn)換成字符。為獲得鏈接System.in的一個(gè)InputStreamReader的對(duì)象,用下面的構(gòu)造函數(shù):
InputStreamReader(InputStream inputStream)
因?yàn)镾ystem .in引用了InputStream 類(lèi)型的對(duì)象,它可以用于inputStream。綜上所述,下面的一行代碼創(chuàng)建了與鍵盤(pán)相連的BufferedReader對(duì)象。
BufferedReader br = new BufferedReader(new
InputStreamReader(System.in));
當(dāng)該語(yǔ)句執(zhí)行后,br是通過(guò)System.in生成的鏈接控制臺(tái)的字符流。
2.1 讀取字符
從BufferedReader讀取字符,用read()。這里所用的read()版本如下:
int read( ) throws IOException
該方法每次執(zhí)行都從輸入流讀取一個(gè)字符然后以整型返回。當(dāng)遇到流的末尾時(shí)它返回-1??梢钥吹?,它要引發(fā)一個(gè)IOException異常。下面的例程演示了read()方法,從控制臺(tái)讀取字符直到用戶(hù)鍵入“q”:
// Use a BufferedReader to read characters from the console.
import java.io.*;
class BRRead {
public static void main(String args[])
throws IOException
{
char c;
BufferedReader br = new
BufferedReader(new InputStreamReader(System.in));
System.out.println("Enter characters, 'q' to quit.");
// read characters
do {
c = (char) br.read();
System.out.println(c);
} while(c != 'q');
}
}
下面是程序運(yùn)行:
Enter characters, 'q' to quit.
123abcq
1
2
3
a
b
c
q
2.2 讀取字符串
從鍵盤(pán)讀取字符串,使用readLine()。它是BufferedReader 類(lèi)的成員。它的通常形式如下:
String readLine( ) throws IOException
它返回一個(gè)String對(duì)象。下面的例子闡述了BufferedReader類(lèi)和readLine()方法;程序讀取和顯示文本的行直到鍵入“stop”:
// Read a string from console using a BufferedReader.
import java.io.*;
class BRReadLines {
public static void main(String args[])
throws IOException
{
// create a BufferedReader using System.in
BufferedReader br = new BufferedReader(new
InputStreamReader(System.in));
String str;
System.out.println("Enter lines of text.");
System.out.println("Enter 'stop' to quit.");
do {
str = br.readLine();
System.out.println(str);
} while(!str.equals("stop"));
}
}
下面的例程生成了一個(gè)小文本編輯器。它創(chuàng)建了一個(gè)String對(duì)象的數(shù)組,然后依行讀取文本,把文本每一行存入數(shù)組。它將讀取到100行或直到按“stop”才停止。該例運(yùn)用一個(gè)BufferedReader類(lèi)來(lái)從控制臺(tái)讀取數(shù)據(jù)。
// A tiny editor.
import java.io.*;
class TinyEdit {
public static void main(String args[])
throws IOException
{
// create a BufferedReader using System.in
BufferedReader br = new BufferedReader(new
InputStreamReader(System.in));
String str[] = new String[100];
System.out.println("Enter lines of text.");
System.out.println("Enter 'stop' to quit.");
for(int i=0; i<100; i++) {
str[i] = br.readLine();
if(str[i].equals("stop")) break;
}
System.out.println("\nHere is your file:");
// display the lines
for(int i=0; i<100; i++) {
if(str[i].equals("stop")) break;
System.out.println(str[i]);
}
}
}
下面是輸出部分:
Enter lines of text.
Enter ‘stop’ to quit.
This is line one.
This is line two.
Java makes working with strings easy.
Just create String objects.
stop
Here is your file:
This is line one.
This is line two.
Java makes working with strings easy.
Just create String objects.
三、向控制臺(tái)寫(xiě)輸出
控制臺(tái)輸出由print( ) 和 println( )來(lái)完成最為簡(jiǎn)單。這兩種方法由PrintStream(System.out引用的對(duì)象類(lèi)型)定義。盡管System.out是一個(gè)字節(jié)流,用它作為簡(jiǎn)單程序的輸出是可行的。因?yàn)镻rintStream是從OutputStream派生的輸出流,它同樣實(shí)現(xiàn)低級(jí)方法write(),write()可用來(lái)向控制臺(tái)寫(xiě)數(shù)據(jù)。PrintStream 定義的write( )的最簡(jiǎn)單的形式如下:
void write(int byteval)
該方法按照byteval指定的數(shù)目向文件寫(xiě)字節(jié)。盡管byteval 定義成整數(shù),但只有低位的8個(gè)字節(jié)被寫(xiě)入。下面的短例用 write()向屏幕輸出字符“A”,然后是新的行。
// Demonstrate System.out.write().
class WriteDemo {
public static void main(String args[]) {
int b;
b = 'A';
System.out.write(b);
System.out.write('\n');
}
}
一般不常用write()來(lái)完成向控制臺(tái)的輸出(盡管這樣做在某些場(chǎng)合非常有用),因?yàn)閜rint()和println() 更容易用。
四、PrintWriter類(lèi)
盡管Java允許用System.out向控制臺(tái)寫(xiě)數(shù)據(jù),但建議僅用在調(diào)試程序時(shí)或在例程中。對(duì)于實(shí)際的程序,Java推薦的向控制臺(tái)寫(xiě)數(shù)據(jù)的方法是用PrintWriter流。PrintWriter是基于字符的類(lèi)。用基于字符類(lèi)向控制臺(tái)寫(xiě)數(shù)據(jù)使程序更為國(guó)際化。PrintWriter定義了多個(gè)構(gòu)造函數(shù),這里所用到的一個(gè)如下:
PrintWriter(OutputStream outputStream, boolean flushOnNewline)
outputStream是OutputStream類(lèi)的對(duì)象,flushOnNewline控制Java是否在println()方法被調(diào)用時(shí)刷新輸出流。如果flushOnNewline為true,刷新自動(dòng)發(fā)生,若為false,則不發(fā)生。
PrintWriter支持所有類(lèi)型(包括Object)的print( )和println( )方法,這樣,就可以像用System.out那樣用這些方法。如果遇到不同類(lèi)型的情況,PrintWriter方法調(diào)用對(duì)象的toString()方法并打印結(jié)果。用PrintWriter向外設(shè)寫(xiě)數(shù)據(jù),指定輸出流為System.out并在每一新行后刷新流。例如這行代碼創(chuàng)建了與控制臺(tái)輸出相連的PrintWriter類(lèi)。
PrintWriter pw = new PrintWriter(System.out, true);
下面的應(yīng)用程序說(shuō)明了用PrintWriter處理控制臺(tái)輸出的方法:
// Demonstrate PrintWriter
import java.io.*;
public class PrintWriterDemo {
public static void main(String args[]) {
PrintWriter pw = new PrintWriter(System.out, true);
pw.println("This is a string");
int i = -7;
pw.println(i);
double d = 4.5e-7;
pw.println(d);
}
}
該程序的輸出如下:
This is a string
-7
4.5E-7
五、文件的讀寫(xiě)
Java提供了一系列的讀寫(xiě)文件的類(lèi)和方法。在Java中,所有的文件都是字節(jié)形式的。Java提供從文件讀寫(xiě)字節(jié)的方法。而且,Java允許在字符形式的對(duì)象中使用字節(jié)文件流。
兩個(gè)最常用的流類(lèi)是FileInputStream和FileOutputStream,它們生成與文件鏈接的字節(jié)流。為打開(kāi)文件,只需創(chuàng)建這些類(lèi)中某一個(gè)類(lèi)的一個(gè)對(duì)象,在構(gòu)造函數(shù)中以參數(shù)形式指定文件的名稱(chēng)。這兩個(gè)類(lèi)都支持其他形式的重載構(gòu)造函數(shù)。下面是這里將要用到的形式:
FileInputStream(String fileName) throws FileNotFoundException
FileOutputStream(String fileName) throws FileNotFoundException
fileName指定需要打開(kāi)的文件名。當(dāng)創(chuàng)建了一個(gè)輸入流而文件不存在時(shí),引發(fā)FileNotFoundException異常。對(duì)于輸出流,如果文件不能生成,則引發(fā)FileNotFoundException異常。如果一個(gè)輸出文件被打開(kāi),所有原先存在的同名的文件被破壞。注意:在早期的Java版本中,當(dāng)輸出文件不能創(chuàng)建時(shí)FileOutputStream()引發(fā)一個(gè)IOException異常。這在Java 2中有所修改。當(dāng)對(duì)文件的操作結(jié)束后,需要調(diào)用close( )來(lái)關(guān)閉文件。該方法在FileInputStream和FileOutputStream中都有定義。如下:
void close( ) throws IOException
為讀文件,可以使用在FileInputStream中定義的read( )方法。這里用到的如下:
int read( ) throws IOException
該方法每次被調(diào)用,它僅從文件中讀取一個(gè)字節(jié)并將該字節(jié)以整數(shù)形式返回。當(dāng)讀到文件尾時(shí),read( )返回-1。該方法可以引發(fā)IOException異常。下面的程序用read()來(lái)輸入和顯示文本文件的內(nèi)容,該文件名以命令行形式指定。注意try/catch塊處理程序運(yùn)行時(shí)可能發(fā)生的兩個(gè)錯(cuò)誤——未找到指定的文件或用戶(hù)忘記包括文件名了。
import java.io.*;
class ShowFile {
public static void main(String args[])
throws IOException
{
int i;
FileInputStream fin;
try {
fin = new FileInputStream(args[0]);
} catch(FileNotFoundException e) {
System.out.println("File Not Found");
return;
} catch(ArrayIndexOutOfBoundsException e) {
System.out.println("Usage: ShowFile File");
return;
}
// read characters until EOF is encountered
do {
i = fin.read();
if(i != -1) System.out.print((char) i);
} while(i != -1);
fin.close();
}
}
向文件中寫(xiě)數(shù)據(jù),需用FileOutputStream定義的write()方法。它的最簡(jiǎn)單形式如下:
void write(int byteval) throws IOException
該方法按照byteval指定的數(shù)目向文件寫(xiě)入字節(jié)。盡管byteval作為整數(shù)聲明,但僅低8位字節(jié)可以寫(xiě)入文件。如果在寫(xiě)的過(guò)程中出現(xiàn)問(wèn)題,一個(gè)IOException被引發(fā)。下面的例子用write()拷貝一個(gè)文本文件:
import java.io.*;
class CopyFile {
public static void main(String args[])
throws IOException
{
int i;
FileInputStream fin;
FileOutputStream fout;
try {
// open input file
try {
fin = new FileInputStream(args[0]);
} catch(FileNotFoundException e) {
System.out.println("Input File Not Found");
return;
}
// open output file
try {
fout = new FileOutputStream(args[1]);
} catch(FileNotFoundException e) {
System.out.println("Error Opening Output File");
return;
}
} catch(ArrayIndexOutOfBoundsException e) {
System.out.println("Usage: CopyFile From To");
return;
}
// Copy File
try {
do {
i = fin.read();
if(i != -1) fout.write(i);
} while(i != -1);
} catch(IOException e) {
System.out.println("File Error");
}
fin.close();
fout.close();
}
}
注意本程序中和前面ShowFile程序中處理潛在輸入/輸出錯(cuò)誤的方法。不像其他的計(jì)算機(jī)語(yǔ)言,包括C和C++,這些語(yǔ)言用錯(cuò)誤代碼報(bào)告文件錯(cuò)誤,而Java用異常處理機(jī)制。這樣不僅是文件處理更為簡(jiǎn)潔,而且使Java正在執(zhí)行輸入時(shí)容易區(qū)分是文件出錯(cuò)還是EOF條件問(wèn)題。在C/C++中,很多輸入函數(shù)在出錯(cuò)時(shí)和到達(dá)文件結(jié)尾時(shí)返回相同的值(也就是說(shuō),在C/C++中,EOF情況與輸入錯(cuò)誤情況映射相同)。這通常意味著程序員必須還要編寫(xiě)特殊程序語(yǔ)句來(lái)判定究竟是哪種事件發(fā)生。Java中,錯(cuò)誤通過(guò)異常引發(fā),而不是通過(guò)read()的返回值。這樣,當(dāng)read( )返回-1時(shí),它僅表示一點(diǎn):遇到了文件的結(jié)尾。