“泛型Java”,一個美麗的hype
今天,Sun發布了J2SE 1.5新特性的一個原型實現版本。這個運行在J2SE 1.4上的插件提供了J2SE 1.5主要的幾項新特性,例如類型安全的枚舉、自動裝箱/拆箱、增強的for循環等,當然還有最受關注的JSR-14,泛型。
在試用了這些新特性之后,我終于對Java的泛型徹底失望了。首先,我們來看看泛型Java的經典用法:
import java.util.*;
?
public class UseGeneric
{
??? public static void main(String[] args)
??? {
??? ??? Vector<Integer> vi = new Vector<Integer>();
??? ??? vi.add(new Integer(24));
??? ??? vi.add(35);
?
??? ??? for(Integer i : vi)
?????? {
?????? ??? System.out.println(i);
?????? }
??? }
}
上面的代碼展示了泛型容器、自動裝箱和增強for循環三項新特性。的確,從簡化代碼的角度來說,這些新特性有一定的幫助——當然,自動裝箱其實不應該算一項很有意義的特性,只是因為Java固有的兩套類型體制將int、char等原生類型與對象區分對待,所以在引入泛型容器時不得不采用自動裝箱作為補救。
將上面的代碼編譯后的class進行反編譯,得到下列代碼:
import java.io.PrintStream;
import java.util.Vector;
?
public class UseGeneric
{
?
??? public UseGeneric()
??? {
??? }
?
??? public static void main(String args[])
??? {
??? ????Vector vector = new Vector();
??????? vector.add(new Integer(24));
??????? vector.add(Integer.valueOf(35));
??????? Integer integer;
??????? for(SimpleIterator simpleiterator = vector.iterator(); simpleiterator.hasNext(); System.out.println(integer))
??????????? integer = (Integer)simpleiterator.next();
?
??? }
}
可以看到,所有的新特性都是在現有虛擬機的基礎上實現的,沒有任何新鮮感可言。的確如Joshua Bloch所說的,只不過是把以前由程序員寫的一些代碼轉成由編譯器來寫。
隨后我試圖實現一些略微高級的泛型技術,例如type traits。我寫了下列代碼:
// General Traits
class NumTraits<T>
{
??? public void doSomething()
??? {
??? ??? System.out.println("General Traits");
??? }
}
?
// Specialized Traits
class NumTraits<Integer)
{
??? public void doSomething()
??? {
??? ??? System.out.println("Traits for Integer");
??? }
}
可惜,這段代碼不能通過編譯,編譯器提示“duplicate class”。顯然,編譯器并沒有把類型參數作為類名稱的一部分,因此traits是不可能實現的了。當然,在成員方法中可以編寫類似于模板特化(specialization)甚至偏特化(partial-specialization)的代碼,但是下面的代碼將證明這種東西毫無意義。
public class Happy<T>
{
??? private T subject = new T();
?
??? public <T> void happy()
??? {
??? ??? subject.beHappy();
??? }
?
??? public static void main(String[] args)
??? {
??? ??? Happy<Dog> o1 = new Happy<Dog>();
??? ??? o1.happy();
?
??? ??? Happy<Cat> o2 = new Happy<Cat>();
??? ??? o2.happy();
??? }
}
這里的編譯錯誤有兩種。首先,“private T subject = new T();”這個語句不能編譯,也許是我還沒有找到實例化類型參數的正確方法吧。更重要的是,編譯器提示“在java.lang.Object中找不到happy()方法”。由于Java采用“擦拭法”實現泛型,所有類型參數(除非顯式聲明超類或接口)都將被擦拭為Object,因此方法調用的契約仍然完全依賴對象系統來保證。換句話說,類似于模板特化之類的技巧不但在效率上毫無幫助,而且根本無法像C++那樣依賴編譯器進行比較高級的檢查甚至編譯期計算。Typelist?Select模板?還是不要想了吧。
喏,這就是所謂的“泛型Java”。沒有編譯期動態綁定,沒有type-traits,沒有(真正的)模板特化,一切的問題依然扔給RTTI來完成。我更愿意把它叫做“Java with some type-safe containers”,而不是“Generic Java”。
泛型Java,一個美麗的hype——如果你對它有太多期望的話。