什麼是注解(Annotation):
Annotation(注解)就是Java提供了一種元程序中的元素關聯 任何信息和著任何元數據(metadata)的途徑和方法。Annotion(注解)是一個接口,程序可以通過反射 來獲取指定程序元素的Annotion對象,然後通過Annotion對象來獲取注解裡面的元數據。
Annotation(注解)是JDK5.0及以後版本引入的。它可以用於創建文檔,跟蹤代碼中的依賴性, 甚至執行基本編譯時檢查。從某些方面看,annotation就像修飾符一樣被使用,並應用於包、類 型、 構造方法、方法、成員變量、參數、本地變量的聲明中。這些信息被存儲在Annotation的“name=value ”結構對中。
Annotation的成員在Annotation類型中以無參數的方法的形式被聲明。其方法名 和返回值定義了該成員的名字和類型。在此有一個特定的默認語法:允許聲明任何Annotation成員的默 認值:一個Annotation可以將name=value對作為沒有定義默認值的Annotation成員的值,當然也可以使 用name=value對來覆蓋其它成員默認值。這一點有些近似類的繼承特性,父類的構造函數可以作為子類 的默認構造函數,但是也可以被子類覆蓋。
Annotation能被用來為某個程序元素(類、方法、 成員變量等)關聯任何的信息。需要注意的是,這裡存在著一個基本的規則:Annotation不能影響程序 代碼的執行,無論增加、刪除 Annotation,代碼都始終如一的執行。另外,盡管一些annotation通過 java的反射api方法在運行時被訪問,而java語言解釋器在工作時忽略了這些annotation。正是由於 java虛擬機忽略了Annotation,導致了annotation類型在代碼中是“不起作用”的; 只有通過某種配 套的工具才會對annotation類型中的信息進行訪問和處理。本文中將涵蓋標准的Annotation和meta- annotation類型,陪伴這些annotation類型的工具是java編譯器(當然要以某種特殊的方式處理它們) 。
什麼是metadata(元數據):
元數據從metadata一詞譯來,就是“關於數據的數據” 的意思。
元數據的功能作用有很多,比如:你可能用過Javadoc的注釋自動生成文檔。這就是元數據 功能的一種。總的來說,元數據可以用來創建文檔,跟蹤代碼的依賴性,執行編譯時格式檢查,代替已 有的配置文件。如果要對於元數據的作用進行分類,目前還沒有明確的定義,不過我們可以根據它所起 的作用,大致可分為三類:p>1. 編寫文檔:通過代碼裡標識的元數據生成文檔p>2. 代碼分析:通過代 碼裡標識的元數據對代碼進行分析p>3. 編譯檢查:通過代碼裡標識的元數據讓編譯器能實現基本的編 譯檢查
在Java中元數據以標簽的形式存在於Java代碼中,元數據標簽的存在並不影響程序代碼的編 譯和執行,它只是被用來生成其它的文件或針在運行時知道被運行代碼的描述信息。
綜上所述:p> 第一,元數據以標簽的形式存在於Java代碼中。p>第二,元數據描述的信息是類型安全的,即元數據內 部的字段都是有明確類型的。p>第三,元數據需要編譯器之外的工具額外的處理用來生成其它的程序部 件。p>第四,元數據可以只存在於Java源代碼級別,也可以存在於編譯之後的Class文件內部。
Annotation和Annotation類型:
Annotation:
Annotation使用了在java5.0所帶來的新 語法,它的行為十分類似public、final這樣的修飾符。每個Annotation具有一個名字和成員個數 >=0。每個Annotation的成員具有被稱為name=value對的名字和值(就像javabean一樣), name=value裝載了Annotation的信息。
Annotation類型:
Annotation類型定義了 Annotation的名字、類型、成員默認值。一個Annotation類型可以說是一個特殊的java接口,它的成員 變量是受限制的,而聲明Annotation類型時需要使用新語法。當我們通過java反射api訪問Annotation 時,返回值將是一個實現了該 annotation類型接口的對象,通過訪問這個對象我們能方便的訪問到其 Annotation成員。後面的章節將提到在java5.0的 java.lang包裡包含的3個標准Annotation類型。
注解的分類:
根據注解參數的個數,我們可以將注解分為三類:p>1.標記注解:一個沒 有成員定義的Annotation類型被稱為標記注解。這種Annotation類型僅使用自身的存在與否來為我們提 供信息。比如後面的系統注解@Override;p>2.單值注解p>3.完整注解
根據注解使用方法和用途 ,我們可以將Annotation分為三類:p>1.JDK內置系統注解p>2.元注解p>3.自定義注解
系統內 置標准注解:
注解的語法比較簡單,除了@符號的使用外,他基本與Java固有的語法一致, JavaSE中內置三個標准注解,定義在java.lang中:p>@Override:用於修飾此方法覆蓋了父類的方 法;p>@Deprecated:用於修飾已經過時的方法;p>@SuppressWarnnings:用於通知java編譯器禁止特定的 編譯警告。
下面我們依次看看三個內置標准注解的作用和使用場景。
@Override,限定 重寫父類方法:
@Override 是一個標記注解類型,它被用作標注方法。它說明了被標注的方法 重載了父類的方法,起到了斷言的作用。如果我們使用了這種Annotation在一個沒有覆蓋父類方法的方 法時,java編譯器將以一個編譯錯誤來警示。這個annotaton常常在我們試圖覆蓋父類方法而確又寫錯 了方法名時發揮威力。使用方法極其簡單:在使用此annotation時只要在被修飾的方法前面加上 @Override即可。下面的代碼是一個使用@Override修飾一個企圖重載父類的displayName()方法,而又 存在拼寫錯誤的實例:
public class Fruit { public void displayName(){ System.out.println("水果的名字是:*****"); } } class Orange extends Fruit { @Override public void displayName(){ System.out.println("水果的名字是:桔子"); } } class Apple extends Fruit { @Override public void displayname(){ System.out.println("水果的名字是:蘋果"); } }
Orange 類編譯不會有任何問題,Apple 類在編譯的時候會提示相應的錯誤。@Override注解 只能用於方法,不能用於其他程序元素。
@Deprecated,標記已過時:
同 樣Deprecated 也是一個標記注解。當一個類型或者類型成員使用@Deprecated修飾的話,編譯器將不鼓勵使用這個被 標注的程序元素。而且這種修飾具有一定的 “延續性”:如果我們在代碼中通過繼承或者覆蓋的方式 使用了這個過時的類型或者成員,雖然繼承或者覆蓋後的類型或者成員並不是被聲明為 @Deprecated, 但編譯器仍然要報警。
值得注意,@Deprecated這個annotation類型和javadoc中的 @deprecated這個tag是有區別的:前者是java編譯器識別的,而後者是被javadoc工具所識別用來生成 文檔(包含程序成員為什麼已經過 時、它應當如何被禁止或者替代的描述)。
在java5.0, java編譯器仍然象其從前版本那樣尋找@deprecated這個javadoc tag,並使用它們產生警告信息。但是 這種狀況將在後續版本中改變,我們應在現在就開始使用@Deprecated來修飾過時的方法而不是 @deprecated javadoc tag。
下面一段程序中使用了@Deprecated注解標示方法過期,同時在方 法注釋中用@deprecated tag 標示該方法已經過時,代碼如下:
class AppleService { public void displayName(){ System.out.println("水果的名字是:蘋果"); } /** * @deprecated 該方法已經過期,不推薦使用 */ @Deprecated public void showTaste(){ System.out.println("水果的蘋果的口感是:脆甜"); } public void showTaste(int typeId){ if(typeId==1){ System.out.println("水果的蘋果的口感是:酸澀"); } else if(typeId==2){ System.out.println("水果的蘋果的口感是:綿甜"); } else{ System.out.println("水果的蘋果的口感是:脆甜"); } } } public class FruitRun { /** * @param args */ public static void main(String[] args) { Apple apple=new Apple(); apple.displayName(); AppleService appleService=new AppleService(); appleService.showTaste(); appleService.showTaste(0); appleService.showTaste(2); } }
AppleService類的showTaste() 方法被@Deprecated標注為過時方法,在FruitRun類中使用 的時候,編譯器會給出該方法已過期,不推薦使用的提示。
SuppressWarnnings,抑制編譯器警 告:
@SuppressWarnings 被用於有選擇的關閉編譯器對類、方法、成員變量、變量初始化的警 告。在java5.0,sun提供的javac編譯器為我們提供了-Xlint選項來使編譯器對合法的程序代碼提出警 告,此種警告從某種程度上代表了程序錯誤。例如當我們使用一個generic collection類而又沒有提供 它的類型時,編譯器將提示出"unchecked warning"的警告。通常當這種情況發生時,我們 就需要查找引起警告的代碼。如果它真的表示錯誤,我們就需要糾正它。例如如果警告信息表明我們代 碼中的switch語句沒有覆蓋所有可能的case,那麼我們就應增加一個默認的case來避免這種警告。
有時我們無法避免這種警告,例如,我們使用必須和非generic的舊代碼交互的generic collection類 時,我們不能避免這個unchecked warning。此時@SuppressWarning就要派上用場了,在調用的方法前 增加@SuppressWarnings修飾,告訴編譯器停止對此方法的警告。
SuppressWarning不是一個標記注 解。它有一個類型為String[]的成員,這個成員的值為被禁止的警告名。對於javac編譯器來講,被- Xlint選項有效的警告 名也同樣對@SuppressWarings有效,同時編譯器忽略掉無法識別的警告名。
annotation語法允許在annotation名後跟括號,括號中是使用逗號分割的name=value對用於為 annotation的成員賦值。實例如下:
public class FruitService { @SuppressWarnings(value={ "rawtypes", "unchecked" }) public static List<Fruit> getFruitList(){ List<Fruit> fruitList=new ArrayList(); return fruitList; } @SuppressWarnings({ "rawtypes", "unchecked" }) public static List<Fruit> getFruit(){ List<Fruit> fruitList=new ArrayList(); return fruitList; } @SuppressWarnings("unused") public static void main(String[] args){ List<String> strList=new ArrayList<String>(); } }
在這個例子中SuppressWarnings annotation類型只定義了一個單一的成員,所以只有一個 簡單的value={...}作為name=value對。又由於成員值是一個數組,故使用大括號來聲明數組值。注意 :我們可以在下面的情況中縮寫annotation:當annotation只有單一成員,並成員命名為 "value="。這時可以省去"value="。比如將上面方法getFruit()的 SuppressWarnings annotation就是縮寫的。
SuppressWarnings注解的常見參數值的簡單說明 :
1.deprecation:使用了不贊成使用的類或方法時的警告;p>2.unchecked:執行了未檢查 的轉換時的警告,例如當使用集合時沒有用泛型 (Generics) 來指定集合保存的類 型;
3.fallthrough:當 Switch 程序塊直接通往下一種情況而沒有 Break 時的警告;p>4.path:在 類路徑、源文件路徑等中有不存在的路徑時的警告; p>5.serial:當在可序列化的類上缺少 serialVersionUID 定義時的警告;
6.finally:任何 finally 子句不能正常完成時的警告;
7.all :關於以上所有情況的警告。