從控制臺(tái)中讀取數(shù)據(jù)是一個(gè)比較常用的功能,在 JDK 5.0 以前的版本中的實(shí)現(xiàn)是比較復(fù)雜的,需要手工處理系統(tǒng)的輸入流。有意思的是,從 JDK 5.0 版本開始,能從控制臺(tái)中輸入數(shù)據(jù)的方法每增加一個(gè)版本號(hào),就有一種新增的方法,這也增加了選擇的種類,可以依據(jù)不同的要求來進(jìn)行選擇。下面來看一下,各個(gè)版本中如何從控制臺(tái)中讀取數(shù)據(jù)以及各自的優(yōu)缺點(diǎn)。
1 JDK 1.4 及以下版本讀取的方法
JDK 1.4 及以下的版本中要想從控制臺(tái)中輸入數(shù)據(jù)只有一種辦法,即使用System.in獲得系統(tǒng)的輸入流,再橋接至字符流從字符流中讀入數(shù)據(jù)。示例代碼如下:
import java.io.IOException;
import java.io.InputStreamReader;
public class Test1 {
public static void main(String[] args) {
String str = readString("請輸入字符串:");
System.out.println("readString 方法的輸入:" + str);
}
/**
* 使用系統(tǒng)的輸入流,從控制臺(tái)中讀取數(shù)據(jù)
* 用于所用的JDK版本
* @param prompt 提示信息
* @return 輸入的字符串
*/
private static String readString(String prompt) {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
String str = null;
try {
System.out.print(prompt);
str = br.readLine();
} catch (IOException e) {
e.printStackTrace();
}
return str;
}
}
從上面的代碼段來看,這種控制臺(tái)輸入的方法非常地麻煩,為了能讀取整行的數(shù)據(jù),采用了BufferedReader類來進(jìn)行處理,而且在讀取的過程中還需要捕獲IOException.不過這是 JDK 1.4 及以下版本中從控制臺(tái)讀取數(shù)據(jù)唯一的辦法。還有一種非控制臺(tái)讀入數(shù)據(jù)的辦法,就是采用 Swing 中的JOptionPane,會(huì)彈出一個(gè)非常漂亮的輸入對話框讓使用者輸入數(shù)據(jù),但這是一種比較另類的做法,不推薦使用。
import javax.swing.JOptionPane;
public class Test2 {
public static void main(String[] args) {
String str = readStringFromDialog("請輸入字符串:");
System.out.println("readStringFromDialog 方法的輸入:" + str);
}
/**
* 使用JOptionPane的輸入對話框,輸入字符串
* 用于所用的JDK版本
* @param prompt 提示信息
* @return 輸入的字符串
*/
private static String readStringFromDialog(String prompt) {
return JOptionPane.showInputDialog(prompt);
}
}
上面的兩種方法都有個(gè)共同的缺點(diǎn)--只能讀取字符串,若需要讀取其他類型的數(shù)據(jù)需要手工進(jìn)行轉(zhuǎn)換。
2 JDK 5.0 讀取的方法
從 JDK 5.0 開始,基本類庫中增加了java.util.Scanner類,根據(jù)它的 API 文檔說明,這個(gè)類是采用正則表達(dá)式進(jìn)行基本類型和字符串分析的文本掃描器。使用它的Scanner(InputStream source)構(gòu)造方法,可以傳入系統(tǒng)的輸入流System.in而從控制臺(tái)中讀取數(shù)據(jù)。示例代碼如下:
import java.util.Scanner;
public class Test3 {
public static void main(String[] args) {
String str = readString5("請輸入字符串:");
System.out.println("readString5 方法的輸入:" + str);
}
/**
* 使用掃描器類(Scanner)從控制臺(tái)中讀取字符串
* 適用于JDK 5.0及以后的版本
* @param prompt 提示信息
* @return 輸入的字符串
*/
private static String readString5(String prompt) {
Scanner scanner = new Scanner(System.in);
System.out.print(prompt);
return scanner.nextLine();
}
}
從代碼量上來看,Test3比Test1少了很多的代碼,核心代碼只有兩行。其實(shí)并不是Scanner將控制臺(tái)輸入給簡單化了,只是在其內(nèi)部的實(shí)現(xiàn)中已經(jīng)將IOException處理了,而且采用InputStreamReader來一個(gè)字符一個(gè)字符進(jìn)行掃描讀取的(嘿嘿,它本身就是個(gè)掃描器),只是Scanner做了更高層次的封裝。
Scanner不僅可以從控制臺(tái)中讀取字符串,還可以讀取除char之外的其他七種基本類型和兩個(gè)大數(shù)字類型,并不需要顯式地進(jìn)行手工轉(zhuǎn)換。Scanner不單單只能掃描控制臺(tái)中輸入的字符,它還可以讓讀入的字符串匹配一定的正則表達(dá)式模式,如果不匹配時(shí)將拋出InputMismatchException異常。
使用System.in作為它的構(gòu)造參數(shù)時(shí),它只掃描了系統(tǒng)輸入流中的字符。它還有其他的構(gòu)造,分別可以從文件或者是字符串中掃描分析字符串的,具體的使用方法可以參考 API 文檔說明。
3 JDK 6.0 讀取的方法
從 JDK 6.0 開始,基本類庫中增加了java.io.Console類,用于獲得與當(dāng)前 Java 虛擬機(jī)關(guān)聯(lián)的基于字符的控制臺(tái)設(shè)備。在純字符的控制臺(tái)界面下,可以更加方便地讀取數(shù)據(jù)。示例代碼如下:
import java.io.Console;
import java.util.Scanner;
public class Test4 {
public static void main(String[] args) {
String str = readString6("請輸入字符串:");
System.out.println("readString6 方法的輸入:" + str);
}
/**
* 使用控制臺(tái)類(Console)從控制臺(tái)中讀取字符串
* 適用于JDK 1.6或以后的版本
* @param prompt 提示信息
* @return 輸入的字符串
*/
private static String readString6(String prompt) {
Console console = System.console();
if (console == null) {
throw new IllegalStateException("不能使用控制臺(tái)");
}
return console.readLine(prompt);
}
}
在Test1和Test3中,輸入數(shù)據(jù)前的提示信息需要使用System.out.print();來輸出,但是使用基于Console的Test4類,可以在方法參數(shù)中直接放入提示信息。
如果需要在控制臺(tái)中輸入密碼等敏感信息的話,像在瀏覽器或者是應(yīng)用程序中那樣顯示替代字符,在 JDK 6.0 以前的做法是相當(dāng)麻煩的(具體的做法可以參考《Java 編程語言中的口令屏蔽》一文),而使用Console類的readPassword()方法可以在控制臺(tái)上不回顯地輸入密碼,并將密碼結(jié)果保存在char數(shù)組中,根據(jù) API 文檔的建議,在使用后應(yīng)立即將數(shù)組清空,以減少其在內(nèi)存中占用的時(shí)間,以便增強(qiáng)安全性。
但是,Console也有一些缺點(diǎn),根據(jù)ConsoleAPI 文檔的說明:
虛擬機(jī)是否具有控制臺(tái)取決于底層平臺(tái),還取決于調(diào)用虛擬機(jī)的方式。如果虛擬機(jī)從一個(gè)交互式命令行開始啟動(dòng),且沒有重定向標(biāo)準(zhǔn)輸入和輸出流,那么其控制臺(tái)將存在,并且通常連接到鍵盤并從虛擬機(jī)啟動(dòng)的地方顯示。如果虛擬機(jī)是自動(dòng)啟動(dòng)的(例如,由后臺(tái)作業(yè)調(diào)度程序啟動(dòng)),那么它通常沒有控制臺(tái)。
通過上面的文檔說明可以看出,在使用 IDE 的情況下,是無法獲取到Console實(shí)例的,原因在于在 IDE 的環(huán)境下,重新定向了標(biāo)準(zhǔn)輸入和輸出流,也是就是將系統(tǒng)控制臺(tái)上的輸入輸出重定向到了 IDE 的控制臺(tái)中。因此,在 IDE 中不能使用這個(gè)程序,而Test1和Test3就沒有這種限制。
4 總結(jié)
以上囊括了 Java 中各種版本從控制臺(tái)中讀入數(shù)據(jù)的方法,將對它們的優(yōu)缺點(diǎn)進(jìn)行了分析。下面給出了一些使用建議,可供參考:
JRE 1.4 或以下版本的情況下,沒得選擇只能采用Test1或者是非控制臺(tái)讀入的Test2的方法。
JRE 5.0 的情況下,建議使用基于Scanner的Test3的方法,更方便地進(jìn)行數(shù)據(jù)讀取。
JRE 6.0 的情況,并且只在字符界面的控制臺(tái)下運(yùn)行時(shí),采用Test4的方法,如果需要讀入像密碼之類的敏感數(shù)據(jù),為了安全性考慮也必須使用Test4或者是自行實(shí)現(xiàn)。如果需要讀入除字符串類型之外的其他數(shù)據(jù)類型,建議使用基于Scanner的控制臺(tái)輸入。
更多信息請查看IT技術(shù)專欄