級(jí)別: 初級(jí) Brett McLaughlin, 作者/編者, O'Reilly Media, Inc
2004 年 9 月 01 日 注
釋?zhuān)琂2SE 5.0 (Tiger) 中的新功能,將非常需要的元數(shù)據(jù)工具引入核心 Java 語(yǔ)言。該系列文章分為兩部分,在這第 1
部分中,作者 Brett McLaughlin 解釋了元數(shù)據(jù)如此有用的原因,向您介紹了 Java 語(yǔ)言中的注釋?zhuān)⒀芯苛?Tiger
的內(nèi)置注釋。
編程的一個(gè)最新的趨勢(shì),尤其是在 Java 編程方面,是使用
元數(shù)據(jù)。簡(jiǎn)單地說(shuō),元數(shù)據(jù)就是
關(guān)于數(shù)據(jù)的數(shù)據(jù)。元數(shù)據(jù)可以用于創(chuàng)建文檔,跟蹤代碼中的依賴(lài)性,甚至執(zhí)行基本編譯時(shí)檢查。許多元數(shù)據(jù)工具,如 XDoclet(請(qǐng)參閱
參考資料),將這些功能添加到核心 Java 語(yǔ)言中,暫時(shí)成為 Java 編程功能的一部分。
直到可以使用 J2SE 5.0(也叫做 Tiger,現(xiàn)在是第二個(gè) beta 版本),核心 Java 語(yǔ)言才最接近具有 Javadoc 方法的元數(shù)據(jù)工具。您使用特殊的標(biāo)簽集合來(lái)標(biāo)記代碼,并執(zhí)行
javadoc
命令來(lái)將這些標(biāo)簽轉(zhuǎn)化成格式化的 HTML 頁(yè)面,該頁(yè)面說(shuō)明標(biāo)簽所附加到的類(lèi)。然而,Javadoc
是有缺陷的元數(shù)據(jù)工具,因?yàn)槌松晌臋n之外,您沒(méi)有固定、實(shí)用、標(biāo)準(zhǔn)化的方式來(lái)將數(shù)據(jù)用于其他用途。HTML 代碼經(jīng)常混入到 Javadoc
輸出中這一事實(shí)甚至更進(jìn)一步降低了其用于任何其他目的的價(jià)值。
Tiger 通過(guò)名為
注釋的新功能將一個(gè)更通用的元數(shù)據(jù)工
具合并到核心 Java 語(yǔ)言中。注釋是可以添加到代碼中的修飾符,可以用于包聲明、類(lèi)型聲明、構(gòu)造函數(shù)、方法、字段、參數(shù)和變量。Tiger
包含內(nèi)置注釋?zhuān)€支持您自己編寫(xiě)的定制注釋。本文將概述元數(shù)據(jù)的優(yōu)點(diǎn)并向您介紹 Tiger 的內(nèi)置注釋。本系列文章的 第 2 部分將研究定制注釋。我要感謝 O'Reilly Media, Inc.,他們非常慷慨地 允許我在本文中使用我關(guān)于 Tiger 的書(shū)籍的“注釋”一章中的代碼示例(請(qǐng)參閱
參考資料)。
元數(shù)據(jù)的價(jià)值
一
般來(lái)說(shuō),元數(shù)據(jù)的好處分為三類(lèi):文檔編制、編譯器檢查和代碼分析。代碼級(jí)文檔最常被引用。元數(shù)據(jù)提供了一種有用的方法來(lái)指明方法是否取決于其他方法,它們
是否完整,特定類(lèi)是否必須引用其他類(lèi),等等。這確實(shí)非常有用,但對(duì)于將元數(shù)據(jù)添加到 Java 語(yǔ)言中來(lái)說(shuō),文檔編制可能是 最不相關(guān)的理由。Javadoc 已經(jīng)提供了非常容易理解和健壯的方法來(lái)文檔化代碼。另外,當(dāng)已經(jīng)存在文檔編制工具,并且在大多數(shù)時(shí)候都工作得很好時(shí),誰(shuí)還要編寫(xiě)文檔編制工具?
編譯器檢查
元數(shù)據(jù)更重要的優(yōu)點(diǎn)是編譯器可以使用它來(lái)執(zhí)行基本的編譯時(shí)檢查。例如,您將在本文后面的
Override 注釋中
看到 Tiger 引入了一個(gè)這樣的注釋?zhuān)糜谠试S您指定一種方法覆蓋超類(lèi)中的另一種方法。Java
編譯器可以確保在元數(shù)據(jù)中指明的行為實(shí)際發(fā)生在代碼級(jí)別。如果從來(lái)沒(méi)有找出過(guò)這種類(lèi)型的 bug,這樣做似乎有點(diǎn)傻,但是大多數(shù)年齡很大的 Java
編程老手都曾經(jīng)花費(fèi)至少多個(gè)晚上來(lái)查明他們的代碼為什么不能用。當(dāng)最后認(rèn)識(shí)到方法的參數(shù)有錯(cuò),且該方法實(shí)際上 沒(méi)有 覆蓋超類(lèi)中的方法時(shí),您可能更感到難受。使用元數(shù)據(jù)的工具有助于輕松地查明這種類(lèi)型的錯(cuò)誤,從而可以節(jié)省那些晚上來(lái)看長(zhǎng)期進(jìn)行的 Halo 聯(lián)賽。
 |
JSR 175
JSR 175,
Java 編程語(yǔ)言的元數(shù)據(jù)工具,為將元數(shù)據(jù)合并到核心 Java 語(yǔ)言中提供了正式理由和說(shuō)明(請(qǐng)參閱
參考資料)。根據(jù) JSR,注釋“不直接影響程序的語(yǔ)義。然而,開(kāi)發(fā)和部署工具可以讀取這些注釋?zhuān)⒁阅撤N形式處理這些注釋?zhuān)赡苌善渌?Java 編程語(yǔ)言源文件、XML 文檔或要與包含注釋的程序一起使用的其他構(gòu)件。”
|
|
代碼分析
可
以證明,任何好的注釋或元數(shù)據(jù)工具的最好功能就是可以使用額外數(shù)據(jù)來(lái)分析代碼。在一個(gè)簡(jiǎn)單的案例中,您可能構(gòu)建代碼目錄,提供必需的輸入類(lèi)型并指明返回類(lèi)
型。但是,您可能想,Java
反射具有相同的優(yōu)點(diǎn);畢竟,可以為所有這些信息內(nèi)省代碼。這從表面上看似乎是正確的,但是在實(shí)際中通常不使用。許多時(shí)候,方法作為輸入接受的或者作為輸出
返回的類(lèi)型實(shí)際上不是該方法想要的類(lèi)型。例如,參數(shù)類(lèi)型可能是 Object ,但方法可能僅使用
Integer 。這在好些情況下很容易發(fā)生,比如在方法被覆蓋而超類(lèi)使用常規(guī)參數(shù)聲明方法時(shí),還有正在進(jìn)行許多序列化的系統(tǒng)中也容易發(fā)生。在這兩種情況中,元數(shù)據(jù)可以指示代碼分析工具,雖然參數(shù)類(lèi)型是
Object ,但
Integer 才是真正需要的。這類(lèi)分析非常有用,但也不能夸大它的價(jià)值。
在
更復(fù)雜的情況下,代碼分析工具可以執(zhí)行所有種類(lèi)的額外任務(wù)。示例 du jour 是 Enterprise JavaBean (EJB)
組件。甚至簡(jiǎn)單 EJB 系統(tǒng)中的依賴(lài)性和復(fù)雜性都非常令人吃驚。您具有了 home 接口和遠(yuǎn)程接口,以及本地接口和本地 home
接口,還有一個(gè)實(shí)現(xiàn)類(lèi)。保持所有這些類(lèi)同步非常困難。但是,元數(shù)據(jù)可以提供這個(gè)問(wèn)題的解決放案。好的工具(還是要提一下
XDoclet)可以管理所有這些依賴(lài)性,并確保無(wú)“代碼級(jí)”連接、但有“邏輯級(jí)”捆綁的類(lèi)保持同步。元數(shù)據(jù)在這里確實(shí)可以發(fā)揮它的作用。
注釋的基本知識(shí)
現(xiàn)在已經(jīng)了解了元數(shù)據(jù)的好處,我將介紹 Tiger 中的注釋。注釋采用“at”標(biāo)記形式 (
@ ),后面是注釋名稱(chēng)。然后在需要數(shù)據(jù)時(shí),通過(guò)
name=value 對(duì)向注釋提供數(shù)據(jù)。每次使用這類(lèi)表示法時(shí),就是在生成注釋。一段代碼可能會(huì)有 10 個(gè)、50 個(gè)或更多的注釋。不過(guò),您將發(fā)現(xiàn)多個(gè)注釋都可能使用相同的
注釋類(lèi)型。類(lèi)型是實(shí)際使用的結(jié)構(gòu),在特定上下文中,注釋本身是該類(lèi)型的具體使用(請(qǐng)參閱側(cè)欄
注釋或注釋類(lèi)型?)。
 |
注釋或注釋類(lèi)型?
是否對(duì)什么是注釋與什么是注釋類(lèi)型感到迷惑?了解這個(gè)的最簡(jiǎn)單方法就是對(duì)比所熟悉的 Java 語(yǔ)言概念來(lái)想。可以定義一個(gè)類(lèi)(例如
Person ),則在 JVM 中將總是僅有該類(lèi)的一個(gè)版本(假設(shè)沒(méi)有進(jìn)行麻煩的類(lèi)路徑設(shè)置)。然而,在任何給定時(shí)間,可能會(huì)使用該類(lèi)的 10 個(gè)或 20 個(gè)
實(shí)例。仍然是只有一個(gè)
Person 類(lèi),但是它以不同的方式使用多次。注釋類(lèi)型和注釋也是這樣。注釋類(lèi)型類(lèi)似于類(lèi),注釋類(lèi)似于該類(lèi)的實(shí)例。
|
|
注釋分為三個(gè)基本種類(lèi):
-
標(biāo)記注釋沒(méi)有變量。注釋顯示簡(jiǎn)單,由名稱(chēng)標(biāo)識(shí),沒(méi)有提供其他數(shù)據(jù)。例如,
@MarkerAnnotation 是標(biāo)記注釋。它不包含數(shù)據(jù),僅有注釋名稱(chēng)。
-
單一值注釋與標(biāo)記注釋類(lèi)似,但提供一段數(shù)據(jù)。因?yàn)閮H提供很少的一點(diǎn)數(shù)據(jù),所以可以使用快捷語(yǔ)法(假設(shè)注釋類(lèi)型接受此語(yǔ)法):
@SingleValueAnnotation("my data") 。除了
@ 標(biāo)記外,這應(yīng)該與普通的 Java 方法調(diào)用很像。
-
完整注釋有多個(gè)數(shù)據(jù)成員。因此,必須使用更完整的語(yǔ)法(注釋不再像普通的 Java 方法):
@FullAnnotation(var1="data value 1", var2="data value 2", var3="data value 3") 。
除了通過(guò)默認(rèn)語(yǔ)法向注釋提供值外,還可以在需要傳送多個(gè)值時(shí)使用名稱(chēng)-值對(duì)。還可以通過(guò)花括號(hào)為注釋變量提供值數(shù)組。清單 1 顯示了注釋中的值數(shù)組的示例。
清單 1. 在注釋中使用按數(shù)組排列的值
@TODOItems({ // Curly braces indicate an array of values is being supplied @TODO( severity=TODO.CRITICAL, item="Add functionality to calculate the mean of the student's grades", assignedTo="Brett McLaughlin" ), @TODO( severity=TODO.IMPOTANT, item="Print usage message to screen if no command-line flags specified", assignedTo="Brett McLaughlin" ), @TODO( severity=TODO.LOW, item="Roll a new website page with this class's new features", assignedTo="Jason Hunter" ) })
|
清單 1 中的示例并沒(méi)有乍一看那樣復(fù)雜。
TODOItems 注釋類(lèi)型有一個(gè)具有值的變量。這里提供的值比較復(fù)雜,但
TODOItems 的使用實(shí)際與單一值注釋類(lèi)型相符,只是這里的單一值是數(shù)組而已。該數(shù)組包含三個(gè)
TODO 注釋?zhuān)渲忻總€(gè)注釋都是多值的。逗號(hào)分隔每個(gè)注釋內(nèi)的值,以及單個(gè)數(shù)組內(nèi)的值。非常容易,是吧?
但是我講的可能超前了些。
TODOItems 和
TODO 是
定制注釋,
是本系列文章第 2 部分中的主題。但是我想讓您看到,即使復(fù)雜注釋?zhuān)ㄇ鍐?1 幾乎是最復(fù)雜的注釋?zhuān)┮膊皇欠浅A钊撕ε碌摹.?dāng)提到 Java
語(yǔ)言的標(biāo)準(zhǔn)注釋類(lèi)型時(shí),將很少看到如此復(fù)雜的情況。正如將在下面三個(gè)部分了解到的,Tiger 的基本注釋類(lèi)型的使用都極其簡(jiǎn)單。
Override 注釋
Tiger 的第一個(gè)內(nèi)置注釋類(lèi)型是
Override 。
Override 應(yīng)該僅用于方法(不用于類(lèi)、包聲明或其他構(gòu)造)。它指明注釋的方法將覆蓋超類(lèi)中的方法。清單 2 顯示了簡(jiǎn)單的示例。
清單 2. 操作中的 Override 注釋
package com.oreilly.tiger.ch06;
public class OverrideTester {
public OverrideTester() { }
@Override public String toString() { return super.toString() + " [Override Tester Implementation]"; }
@Override public int hashCode() { return toString().hashCode(); } }
|
清單 2 應(yīng)該很容易理解。
@Override 注釋對(duì)兩個(gè)方法進(jìn)行了注釋 —
toString() 和
hashCode() ,來(lái)指明它們覆蓋
OverrideTester 類(lèi)的超類(lèi) (
java.lang.Object ) 中的方法的版本。開(kāi)始這可能看起來(lái)沒(méi)什么作用,但它實(shí)際上是非常好的功能。如果不覆蓋這些方法,根本
無(wú)法 編譯此類(lèi)。該注釋還確保當(dāng)您將
toString() 弄亂時(shí),至少還有某種指示,即應(yīng)該確保
hashCode() 仍舊匹配。
當(dāng)編碼到很晚且輸錯(cuò)了某些東西時(shí),此注釋類(lèi)型真的可以發(fā)揮很大的作用,如清單 3 中所示。
清單 3. 使 Override 注釋捕獲打字稿
package com.oreilly.tiger.ch06;
public class OverrideTester {
public OverrideTester() { }
@Override public String toString() { return super.toString() + " [Override Tester Implementation]"; }
@Override public int hasCode() { return toString().hashCode(); } }
|
在清單 3 中,
hashCode() 錯(cuò)誤地輸入為
hasCode() 。注釋指明
hasCode() 應(yīng)該覆蓋方法。但是在編譯中,
javac 將發(fā)現(xiàn)超類(lèi) (
java.lang.Object ) 沒(méi)有名為
hasCode() 的方法可以覆蓋。因此,編譯器將報(bào)錯(cuò),如圖 1 中所示。
圖 1. 來(lái)自 Override 注釋的編譯器警告
 |
缺少的功能
在單一值注釋類(lèi)型中,如果
Deprecated 允許包含錯(cuò)誤類(lèi)型消息將更好。然后,當(dāng)用戶(hù)使用聲明為過(guò)時(shí)的方法時(shí),編譯器可以打印消息。該消息可以指明使用方法的結(jié)果如何重要,說(shuō)明何時(shí)將停止方法,甚至建議備用方法。可能 J2SE 的下一版本(“Mustang”,他們這樣命名)將提供這種功能。
|
|
這個(gè)便捷的小功能將幫助快速捕獲打字稿。
Deprecated 注釋
下一個(gè)標(biāo)準(zhǔn)注釋類(lèi)型是
Deprecated 。與
Override 一樣,
Deprecated 是標(biāo)記注釋。正如您可能期望的,可以使用
Deprecated 來(lái)對(duì)不應(yīng)再使用的方法進(jìn)行注釋。與
Override 不一樣的是,
Deprecated 應(yīng)該與正在聲明為過(guò)時(shí)的方法放在同一行中(為什么這樣?說(shuō)實(shí)話(huà)我也不知道),如清單 4 中所示。
清單 4. 使用 Deprecated 注釋
package com.oreilly.tiger.ch06;
public class DeprecatedClass {
@Deprecated public void doSomething() { // some code }
public void doSomethingElse() { // This method presumably does what doSomething() does, but better } }
|
單獨(dú)編譯此類(lèi)時(shí),不會(huì)發(fā)生任何不同。但是如果通過(guò)覆蓋或調(diào)用來(lái)使用聲明為過(guò)時(shí)的方法,編譯器將處理注釋?zhuān)l(fā)現(xiàn)不應(yīng)該使用該方法,并發(fā)出錯(cuò)誤消息,如圖 2 中所示。
圖 2. 來(lái)自 Deprecated 注釋的編譯器警告
注意需要開(kāi)啟編譯器警告,就像是必須向 Java 編譯器指明想要普通的聲明為過(guò)時(shí)警告。可以使用下列兩個(gè)標(biāo)記之一和
javac 命令:
-deprecated 或新的
-Xlint:deprecated 標(biāo)記。
SuppressWarnings 注釋
從 Tiger “免費(fèi)”獲得的最后一個(gè)注釋類(lèi)型是
SuppressWarnings 。發(fā)現(xiàn)該類(lèi)型的作用應(yīng)該不難,但是
為什么該注釋類(lèi)型如此重要通常不是很明顯。它實(shí)際上是 Tiger 的所有新功能的副功能。例如,以泛型為例;泛型使所有種類(lèi)的新類(lèi)型安全操作成為可能,特別是當(dāng)涉及 Java 集合時(shí)。然而,因?yàn)榉盒停?dāng)使用集合而
沒(méi)有 類(lèi)型安全時(shí),編譯器將拋出警告。這對(duì)于針對(duì) Tiger 的代碼有幫助,但它使得為 Java 1.4.x 或更早版本編寫(xiě)代碼非常麻煩。將不斷地收到關(guān)于根本無(wú)關(guān)的事情的警告。如何才能使編譯器不給您增添麻煩?
SupressWarnings 可以解決這個(gè)問(wèn)題。
SupressWarnings 與
Override 和
Deprecated 不同,
是具有變量的 — 所以您將單一注釋類(lèi)型與該變量一起使用。可以以值數(shù)組來(lái)提供變量,其中每個(gè)值指明要阻止的一種特定警告類(lèi)型。請(qǐng)看清單 5 中的示例,這是 Tiger 中通常會(huì)產(chǎn)生錯(cuò)誤的一些代碼。
清單 5. 不是類(lèi)型安全的 Tiger 代碼
public void nonGenericsMethod() { List wordList = new ArrayList(); // no typing information on the List
wordList.add("foo"); // causes error on list addition }
|
圖 3 顯示了清單 5 中代碼的編譯結(jié)果。
圖 3. 來(lái)自非標(biāo)準(zhǔn)代碼的編譯器警告
清單 6 通過(guò)使用
SuppressWarnings 注釋消除了這種問(wèn)題。
清單 6. 阻止警告
@SuppressWarings(value={"unchecked"}) public void nonGenericsMethod() { List wordList = new ArrayList(); // no typing information on the List
wordList.add("foo"); // causes error on list addition }
|
非常簡(jiǎn)單,是吧??jī)H需要找到警告類(lèi)型(圖 3 中顯示為“unchecked”),并將其傳送到
SuppressWarnings 中。
SuppressWarnings 中變量的值采用數(shù)組,使您可以在同一注釋中阻止多個(gè)警告。例如,
@SuppressWarnings(value={"unchecked", "fallthrough"}) 使用兩個(gè)值的數(shù)組。此功能為處理錯(cuò)誤提供了非常靈活的方法,無(wú)需進(jìn)行大量的工作。
結(jié)束語(yǔ)
雖
然這里看到的語(yǔ)法可能都是新的,但您應(yīng)該知道注釋非常容易理解和使用。也就是說(shuō),與 Tiger
一起提供的標(biāo)準(zhǔn)注釋相當(dāng)簡(jiǎn)單,可以添加許多功能。元數(shù)據(jù)正日益變得有幫助,您肯定會(huì)提出非常適用于自己的應(yīng)用程序的注釋類(lèi)型。在本系列文章的第 2
部分,我將詳細(xì)說(shuō)明 Tiger 對(duì)編寫(xiě)自己的注釋類(lèi)型的支持。您將了解如何創(chuàng)建 Java
類(lèi)以及將其定義為注釋類(lèi)型,如何使編譯器識(shí)別您的注釋類(lèi)型,以及如何使用該類(lèi)型對(duì)代碼進(jìn)行注釋。我甚至?xí)钊氲卣f(shuō)明奇異但有用的對(duì)注釋進(jìn)行注釋的任務(wù)。
您將快速熟悉 Tiger 中的這一新構(gòu)造。
參考資料
關(guān)于作者  | 
| Brett
McLaughlin 從 Logo 時(shí)代(還記得那個(gè)小三角嗎?)就開(kāi)始從事計(jì)算機(jī)工作。在最近幾年里,他已經(jīng)成為 Java 和 XML
社區(qū)最知名的作者和程序員之一。他曾經(jīng)在 Nextel Communications 實(shí)現(xiàn)復(fù)雜的企業(yè)系統(tǒng),在 Lutris
Technologies 編寫(xiě)應(yīng)用程序服務(wù)器,目前在為 O'Reilly Media, Inc 撰寫(xiě)和編輯 書(shū)籍。
|
|