JDK1.5中新的語言特徵
本文中,我們將討論幾個在JDK1.5中新的語言特徵,包括:
泛型(Generics)--為集合(collections)提供編譯時型別安全,無需每刻從Collections取得一個物件就進行強制轉換(cast)
增強的“for”迴圈(Enhanced For loop)--減少迭代器(iterator)的潛在錯誤(error-proneness)
自動置入/自動取出(Autoboxing/unboxing)--無需在基本型別(primitive types)(例如double)和包裝型別(wrapper types)(例如Double)之間人工地進行轉換。
型別安全的列舉(Typesafeenums)--提供型別安全列舉模式的各項好處。
靜態匯入(Static import)--無需在使用其他類的靜態成員變數字首其類名.這將使得程式碼更為簡潔。
後設資料(Metadata)--使程式設計人員避免編寫樣板化程式碼(boiler plate code),並提供機會進行宣告式程式設計(declarative programming)。
讓我們詳細討論每個新特性,並看一些例子。
泛型是JDK1.5中一個最“酷”的特徵。通過引入泛型,我們將獲得編譯時型別的安全和執行時更小地丟擲ClassCastExceptions的可能。在JDK1.5中,你可以宣告一個集合將接收/返回的物件的型別。在JDK1.4中,建立僱員名字的清單(List)需要一個集合物件,像下面的語句:
List listOfEmployeeName = new ArrayList();
在JDK1.5中,你將使用下面語句
List<String> listOfEmployeeName = new ArrayList<String>();
最“酷”的是,如果你試圖插入非string型別的值,你將在編譯時發現並且修正這類問題。沒有泛型,你會發現這樣一個bug,當你的客戶呼叫後會告訴你,你所編寫的程式丟擲ClassCastException異常而崩潰。
另外,當你從集合中得到一個元素時你無需進行強制轉換。故原先為:
String employeeName = ((String) listOfEmployee.get(i));
而下面的語句將比上面的更加簡單:
String employeeName = listOfEmployee.get(i);
不清楚物件的型別而強制轉換物件是不合理的,並且更重要的是,它將在執行時失敗。假使使用者無意間傳入一個包含string buffers型別而非string型別的集合,那結果會怎樣呢。在Listing A中,客戶被要求傳入一個編譯器無法強制的strings型別集合。Listing B中顯示了同樣的方法使用泛型是如何實現的。
Listing A
staticbooleancheckName(Collection employeeNameList, String name) {
for (Iteratori = employeeNamList.iterator(); i.hasNext(); ) {
String s = (String) i.next();
if(s.equals(name)){
return true;
//print employee name here ......
}
}
return false;
}
Listing B
staticbooleancheckName(Collection<String> employeeNameList, String name) {
for (Iteratori = employeeNamList.iterator(); i.hasNext(); ) {
if(i.next().equals(name)){
return true;
//print employee name here ......
}
}
return false;
}
現在,通過方法簽名可以清楚知道輸入集合必須只能包含strings。如果客戶試圖傳入一個包含string buffers的集合,程式將不會編譯。同時注意,該方法不包含任何強制轉換。它只需要短短一行,一旦你習慣泛型後,它也更加清晰。
在JDK當前版本下的For迴圈語法如下:
void printAll(Collection c) {
for (Iteratori = c.iterator(); i.hasNext(); ) {
Employee emp = (Employee)i.next();
System.out.println(emp.getName());
}
}
現在,用增強的For語句實現相同方法:
voidprintAll(Collection c) {
for (Object o : c)
System.out.println((TimerTask)o).getName());
}
在這類For迴圈中,你應該將":"看成"in",所以,在該例中可以看成"for Object o in c"。你可以發現這種For迴圈更具可讀性。
自動置入/自動取出(Autoboxing/unboxing)
Java有基本資料型別,在這些基本資料型別周圍又有包裝類。通常,程式設計人員需要將一種型別轉換成另一種。看看Listing C.中的程式碼片斷。
Listing C
public class Employee {
private static final Integer CHILD = new Integer(0);
public static void main(String args[]) {
//code for adding n to an Integer
int n=10;
Integer age= new Integer(30);
Integer ageAfterTenYear= new Integer(age.intValue +10);
}
}
請注意,用於計算ageAfterTenYear的內迴圈程式碼看上去是多麼雜亂。現在,在Listing D.中看看相同的程式使用autoboxing重寫後的樣子。
Listing D
public class Employee {
public static void main(String args[]) {
int n=10;
Integer age= new Integer(30);
Integer ageAfterTenYear= age +10;
}
}
有一件事值得注意的:在先前,如果你取出(unbox)Null值,它將變為0。在次程式碼中,編譯器將自動地轉換Integer為int然後加上10,接著將其轉換回Integer.。
型別安全的列舉(Typesafeenums)
型別安全列舉提供下列特性:
他們提供編譯時型別安全。
他們都是物件,因此你不需要將他們放入集合中。
他們作為一種類的實現,因此你可以新增一些方法。
他們為列舉型別提供了合適的名稱空間。
他們列印的值具有情報性(informative)? 如果你列印一個整數列舉(intenum),你只是看見一個數字,它可能並不具有情報性。
例一:
enum Season { winter, spring, summer, fall }
例二:
public enum Coin {
penny(1), nickel(5), dime(10), quarter(25);
Coin(int value) { this.value = value; }
private final int value;
public int value() { return value; }
}
靜態匯入(Static import)
靜態匯入使程式碼更易讀。通常,你要使用定義在另一個類中的常量(constants),像這樣:
importorg.yyy.pkg.Increment;
class Employee {
public Double calculateSalary(Double salary{
return salary + Increment.INCREMENT * salary;
}
}
當時使用靜態匯入,我們無需為常量名字首類名就能使用這些常量,像這樣:
import static org.yyy.pkg.Increment;
class Employee {
public Double calculateSalary(Double salary{
return salary + INCREMENT * salary;
}
}
注意,我們可以呼叫INCREMENT這一常量而不要使用類名Increment.。
後設資料(Metadata)
後設資料特徵志於使開發者們藉助廠商提供的工具可以進行更簡易的開發。看一看Listing E.中的程式碼。
Listing E
importorg.yyy.hr;
public interface EmployeeI extends Java.rmi.Remote {
public String getName()
throwsJava.rmi.RemoteException;
public String getLocation ()
throwsJava.rmi.RemoteException;
}
public class EmployeeImpl implements EmployeeI {
public String getName(){
}
public String getLocation (){
}
}
通過後設資料的支援,你可以改寫Listing E中的程式碼為:
importorg.yyy.hr;
public class Employee {
@Remote public String getName() {
...
}
@Remote public public String getLocation() {
...
}
}
正像你所看到的,所有樣板化的程式碼都不見了。
這些新特性和規格說明將在JDK1.5中實現。它將提供Java程式設計社群更多的選擇以編寫魯棒的、可擴充套件的程式碼。認真的Java程式設計人員將感到著手去熟悉這一Java程式語言的新版本是很有益的
泛型(Generics)--為集合(collections)提供編譯時型別安全,無需每刻從Collections取得一個物件就進行強制轉換(cast)
增強的“for”迴圈(Enhanced For loop)--減少迭代器(iterator)的潛在錯誤(error-proneness)
自動置入/自動取出(Autoboxing/unboxing)--無需在基本型別(primitive types)(例如double)和包裝型別(wrapper types)(例如Double)之間人工地進行轉換。
型別安全的列舉(Typesafeenums)--提供型別安全列舉模式的各項好處。
靜態匯入(Static import)--無需在使用其他類的靜態成員變數字首其類名.這將使得程式碼更為簡潔。
後設資料(Metadata)--使程式設計人員避免編寫樣板化程式碼(boiler plate code),並提供機會進行宣告式程式設計(declarative programming)。
讓我們詳細討論每個新特性,並看一些例子。
泛型(Generics)
泛型是JDK1.5中一個最“酷”的特徵。通過引入泛型,我們將獲得編譯時型別的安全和執行時更小地丟擲ClassCastExceptions的可能。在JDK1.5中,你可以宣告一個集合將接收/返回的物件的型別。在JDK1.4中,建立僱員名字的清單(List)需要一個集合物件,像下面的語句:
List listOfEmployeeName = new ArrayList();
在JDK1.5中,你將使用下面語句
List<String> listOfEmployeeName = new ArrayList<String>();
最“酷”的是,如果你試圖插入非string型別的值,你將在編譯時發現並且修正這類問題。沒有泛型,你會發現這樣一個bug,當你的客戶呼叫後會告訴你,你所編寫的程式丟擲ClassCastException異常而崩潰。
另外,當你從集合中得到一個元素時你無需進行強制轉換。故原先為:
String employeeName = ((String) listOfEmployee.get(i));
而下面的語句將比上面的更加簡單:
String employeeName = listOfEmployee.get(i);
不清楚物件的型別而強制轉換物件是不合理的,並且更重要的是,它將在執行時失敗。假使使用者無意間傳入一個包含string buffers型別而非string型別的集合,那結果會怎樣呢。在Listing A中,客戶被要求傳入一個編譯器無法強制的strings型別集合。Listing B中顯示了同樣的方法使用泛型是如何實現的。
Listing A
staticbooleancheckName(Collection employeeNameList, String name) {
for (Iteratori = employeeNamList.iterator(); i.hasNext(); ) {
String s = (String) i.next();
if(s.equals(name)){
return true;
//print employee name here ......
}
}
return false;
}
Listing B
staticbooleancheckName(Collection<String> employeeNameList, String name) {
for (Iteratori = employeeNamList.iterator(); i.hasNext(); ) {
if(i.next().equals(name)){
return true;
//print employee name here ......
}
}
return false;
}
現在,通過方法簽名可以清楚知道輸入集合必須只能包含strings。如果客戶試圖傳入一個包含string buffers的集合,程式將不會編譯。同時注意,該方法不包含任何強制轉換。它只需要短短一行,一旦你習慣泛型後,它也更加清晰。
在JDK當前版本下的For迴圈語法如下:
void printAll(Collection c) {
for (Iteratori = c.iterator(); i.hasNext(); ) {
Employee emp = (Employee)i.next();
System.out.println(emp.getName());
}
}
現在,用增強的For語句實現相同方法:
voidprintAll(Collection c) {
for (Object o : c)
System.out.println((TimerTask)o).getName());
}
在這類For迴圈中,你應該將":"看成"in",所以,在該例中可以看成"for Object o in c"。你可以發現這種For迴圈更具可讀性。
自動置入/自動取出(Autoboxing/unboxing)
Java有基本資料型別,在這些基本資料型別周圍又有包裝類。通常,程式設計人員需要將一種型別轉換成另一種。看看Listing C.中的程式碼片斷。
Listing C
public class Employee {
private static final Integer CHILD = new Integer(0);
public static void main(String args[]) {
//code for adding n to an Integer
int n=10;
Integer age= new Integer(30);
Integer ageAfterTenYear= new Integer(age.intValue +10);
}
}
請注意,用於計算ageAfterTenYear的內迴圈程式碼看上去是多麼雜亂。現在,在Listing D.中看看相同的程式使用autoboxing重寫後的樣子。
Listing D
public class Employee {
public static void main(String args[]) {
int n=10;
Integer age= new Integer(30);
Integer ageAfterTenYear= age +10;
}
}
有一件事值得注意的:在先前,如果你取出(unbox)Null值,它將變為0。在次程式碼中,編譯器將自動地轉換Integer為int然後加上10,接著將其轉換回Integer.。
型別安全的列舉(Typesafeenums)
型別安全列舉提供下列特性:
他們提供編譯時型別安全。
他們都是物件,因此你不需要將他們放入集合中。
他們作為一種類的實現,因此你可以新增一些方法。
他們為列舉型別提供了合適的名稱空間。
他們列印的值具有情報性(informative)? 如果你列印一個整數列舉(intenum),你只是看見一個數字,它可能並不具有情報性。
例一:
enum Season { winter, spring, summer, fall }
例二:
public enum Coin {
penny(1), nickel(5), dime(10), quarter(25);
Coin(int value) { this.value = value; }
private final int value;
public int value() { return value; }
}
靜態匯入(Static import)
靜態匯入使程式碼更易讀。通常,你要使用定義在另一個類中的常量(constants),像這樣:
importorg.yyy.pkg.Increment;
class Employee {
public Double calculateSalary(Double salary{
return salary + Increment.INCREMENT * salary;
}
}
當時使用靜態匯入,我們無需為常量名字首類名就能使用這些常量,像這樣:
import static org.yyy.pkg.Increment;
class Employee {
public Double calculateSalary(Double salary{
return salary + INCREMENT * salary;
}
}
注意,我們可以呼叫INCREMENT這一常量而不要使用類名Increment.。
後設資料(Metadata)
後設資料特徵志於使開發者們藉助廠商提供的工具可以進行更簡易的開發。看一看Listing E.中的程式碼。
Listing E
importorg.yyy.hr;
public interface EmployeeI extends Java.rmi.Remote {
public String getName()
throwsJava.rmi.RemoteException;
public String getLocation ()
throwsJava.rmi.RemoteException;
}
public class EmployeeImpl implements EmployeeI {
public String getName(){
}
public String getLocation (){
}
}
通過後設資料的支援,你可以改寫Listing E中的程式碼為:
importorg.yyy.hr;
public class Employee {
@Remote public String getName() {
...
}
@Remote public public String getLocation() {
...
}
}
正像你所看到的,所有樣板化的程式碼都不見了。
這些新特性和規格說明將在JDK1.5中實現。它將提供Java程式設計社群更多的選擇以編寫魯棒的、可擴充套件的程式碼。認真的Java程式設計人員將感到著手去熟悉這一Java程式語言的新版本是很有益的
相關文章
- 各種語言按鈕事件特徵碼事件特徵
- Python程式語言有哪些特徵?入門分享!Python特徵
- fpcx教程:在Final Cut Pro中新增多語言字幕
- 語音的關鍵聲學特徵(語音情感特徵提取)特徵
- Python語言具有哪些優秀特徵?為什麼受歡迎?Python特徵
- 如何結合文字特徵檢測仇恨和攻擊性語言特徵
- Java JDK1.5: 泛型 新特性的講解說明JavaJDK泛型
- go語言與c語言的相互呼叫GoC語言
- 計算機程式語言的分類,解釋型語言、編譯型語言、指令碼語言的關係計算機編譯指令碼
- Python學習中:最感到驚奇35個語言特徵和程式設計技巧Python特徵程式設計
- Go是Google的語言,而不是我們的語言Go
- C語言:一種高效、易學的程式語言C語言
- Go語言的”坑“Go
- go語言的介面Go
- 那些主流程式語言的知識,C語言(Ⅰ)C語言
- C語言與嵌入式C語言的區別C語言
- 語言型別介紹及其Python的語言型別型別Python
- 存在可能替代C語言的其他語言嗎? -Erik EngheimC語言
- 計算機語言:編譯型/解釋型、動態語言/靜態語言、強型別語言/弱型別語言計算機編譯型別
- 中文程式語言——易語言,到底是用來幹什麼的?易語言值得學習嗎?易語言的優勢有什麼?
- Go語言————1、初識GO語言Go
- java語言屬於哪種語言Java
- 編譯語言、解釋語言與指令碼語言之間的區別編譯指令碼
- 自然語言處理中的語言模型預訓練方法自然語言處理模型
- nlp中的傳統語言模型與神經語言模型模型
- 思想止於語言?語言只是思想的腳手架 - Nautilus
- 概念區別 【編譯型語言與解釋型語言、動態型別語言與靜態型別語言、強型別語言與弱型別語言】編譯型別
- 【Go語言入門系列】(八)Go語言是不是面嚮物件語言?Go物件
- Go語言版本的forgeryGo
- c語言巨集的使用C語言
- Go語言的前景分析Go
- 有用的C語言工具C語言
- Go語言的那些坑Go
- c語言的基礎C語言
- Laravel中的多語言Laravel
- 程式語言的變遷
- C語言的函式C語言函式
- c語言的陣列C語言陣列
- SQL語言(結構化查詢語言)SQL