原文位址:http://blog.sina.com.cn/s/blog_6f6a95180100q0tv.html
辨別接口是沒有任何方法和屬性的接口。辨別接口不對實作它的類有任何語義上的要求,它僅僅表明實作它的類屬于一個特定的類型。
标接口在Java語言中有一些很著名的應用,比如java.io.Serializable和java.rmi.Remote等接口便是辨別接口。辨別接口,當一個類實作了一個辨別接口之後就像是給自己打了個标簽。
為此,我們通過一個通俗而有趣的示例!這個示例是設計一個獵人,其持有一把智能獵槍,這就是說這把獵槍會自動識别人類,若發現瞄準的目标是人類,就不會開火,而其它的任何事物都通殺。
為此,我們使用了下面三個接口:
一個用來表示萬事萬物的SomeThing
public
interface SomeThing {}
人類的接口:
public interface Humans extendsSomeThing
{}
動物的接口:
public interface Animals extendsSomeThing
{}
然後是一系列的實作:
中國人:
public
class Chinese implements Humans {}
日本人:
public
class Japanese {}
狗:
public class Dog implements Animals {}
妖怪(他很聰明,給自己帖上了人的标簽):
public class Monsterimplements
Humans {}
下面這個程式的核心部分,獵人類及用戶端程式:
public class Hunter{
public
void fire(Object target)
{
if(target instanceof Humans)
{
System.out.println("這下完了,打中了一個人,該去坐牢了!");
}
else
{
System.out.println("恭喜你,打中了一隻動物!");
}
}
//智能的槍
public void
intelligentFire(Object target)
{
if(target instanceof Humans)
{
return;
}
System.out.println("開了一槍!"+target.getClass());
//下面進行秒殺等相關處理
//銷毀他
target=null;
}
public
static void main(String[] args) {
Hunter hunter=new Hunter();
Object[]
objects=new Object[]{new Dog(),new Japanese(),new Japanese(),newChinese(),new
Monster(),new SomeThing(){}};
for(int
i=0;i<objects.length;i++)
hunter.intelligentFire(objects[i]);
}
}
運作程式,你會發現輸出類似下面結果:
開了一槍!class springroad.demo.taginterface.Dog
開了一槍!class
springroad.demo.taginterface.Japanese
開了一槍!class
springroad.demo.taginterface.Japanese
開了一槍!class
springroad.demo.taginterface.Hunter$1
由此可見,智能獵槍瞄準6個目标,開了4槍。隻對Chinese、及Monster的執行個體沒有開槍。因為這裡讨論的是标簽接口,雖然Humans沒有任何方法,但從智能獵槍的角度來看,他就是通過這個标簽來判斷是否可以開火的。他不用管也管不了目标的層次等級關系(比如Japanese肯定很非常鮮明等級結構),即繼承關系。他也管不了目标的來自于哪兒。比如,是用new操作符建立,還是從容器中取,或者是從網絡某個地方加載一個。
Hunter隻是制訂了一個簡單的規則,你要想不讓我的槍對你開火,你就必須在自己身上帖上一個Humans的标簽。也就是說你必須遵守這個規則。
現在回過頭來看,因為妖怪Monster真身應該是一條蛇或其它什麼動物,但是他懂得Hunter制訂的規則,于在巧妙的給自己帖上了一個Humans的标簽,以緻于欺騙了我們的智能獵槍。
而Japanese則自認為自己了不起,不按規則辦事,我就不理你Hunter制訂的規則,什麼Humans标簽,我就是不用。于是放到我們的程式中當然就隻有挨殺的份了。
由此可見,空接口(标簽接口)的重要性,在像本例中,給不給自己帖上标簽,這是一個性命莜關的問題。其實在OO的世界中,空接口可以算是最高的層像。