1. 程式人生 > 實用技巧 >什麼是Java Marker Interface(標記介面)

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

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

首先明確一點,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。

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

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

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

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



作者:JerryWang_汪子熙
連結:https://www.jianshu.com/p/40c6e5cd48b0
來源:簡書
著作權歸作者所有。商業轉載請聯絡作者獲得授權,非商業轉載請註明出處。