什麼是Java Marker Interface(標記介面)

i042416發表於2018-08-29

先看看什麼是標記介面?標記介面有時也叫標籤介面(Tag interface),即介面不包含任何方法。在Java裡很容易找到標記介面的例子,比如JDK裡的Serializable介面就是一個標記介面。

什麼是Java Marker Interface(標記介面)

首先明確一點,Marker Interface(標記介面)決不是Java這門程式語言特有的,而是電腦科學中一種通用的設計理念。

我們看Wikipedia裡對標記介面的定義。

“The tag/ marker interface pattern is a design pattern in computer science, used with languages that provide run-time type information about objects. It provides a means to associate metadata with a class where the language does not have explicit support for such metadata.“

我試了下Google Translate翻譯上面這段話,翻得很差勁,所以我來解釋一下。

標記介面是電腦科學中的一種設計思路。程式語言本身不支援為類維護後設資料。而標記介面則彌補了這個功能上的缺失——一個類實現某個沒有任何方法的標記介面,實際上標記介面從某種意義上說就成為了這個類的後設資料之一。執行時,透過程式語言的反射機制,我們就可以在程式碼裡拿到這種後設資料。

以Serializable介面為例。一個類實現了這個介面,說明它可以被序列化。因此,我們實際上透過Serializable這個介面,給該類標記了“可被序列化”的後設資料,打上了“可被序列化”的標籤。這也是標記/標籤介面名字的由來。

下面的程式碼是我從JDK原始碼中摘出來的:

if (obj instanceof String) {
    writeString((String) obj, unshared);
} else if (cl.isArray()) {
    writeArray(obj, desc, unshared);
} else if (obj instanceof Enum) {
    writeEnum((Enum) obj, desc, unshared);
} else if (obj instanceof Serializable) {
    writeOrdinaryObject(obj, desc, unshared);
} else {    if (extendedDebugInfo) {        throw new NotSerializableException(cl.getName() + " "
        + debugInfoStack.toString());
    } else {        throw new NotSerializableException(cl.getName());
    }
}

Java裡的序列化,字串,陣列,列舉類和普通類是分別進行的。如果當前待序列化的變數既不是字串,也不是陣列和列舉類,那麼就檢測該類是否實現了Serializable的介面,大家注意下圖第1177行就執行了這種檢測。如果沒有實現Serializable介面,就會丟擲異常NotSerializableException。

什麼是Java Marker Interface(標記介面)

大家也許會問,在Spring裡滿天飛的註解(Annotation)不是最好的用來維護後設資料的方式麼?確實,Annotation能宣告在Java包、類、欄位、方法、區域性變數、方法引數等的前面用於維護後設資料的目的,既靈活又方便。然而這麼好的東西,只有在JDK1.5之後才能用。JDK1.5之前維護後設資料的重任就落在標記介面上了。

大家看另一個標記介面,Cloneable。下圖第51行清晰標註了該介面從JDK1.0起就有了。

什麼是Java Marker Interface(標記介面)

JDK原始碼裡的Clone方法的註釋也清晰註明了,如果一個類沒有實現Cloneable介面,在執行clone方法時會丟擲CloneNotSupportedException異常。

什麼是Java Marker Interface(標記介面)

要獲取更多Jerry的原創技術文章,請關注公眾號"汪子熙"或者掃描下面二維碼:


什麼是Java Marker Interface(標記介面)

什麼是Java Marker Interface(標記介面)


來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/24475491/viewspace-2213091/,如需轉載,請註明出處,否則將追究法律責任。

相關文章