我們在JAVA中處處都用到了范型,JAVA中的范型是從C++模板繼承來的,不過Java的范型的功能遠遠沒有C++那麼強大。
我們知道在C++中模板可以很方便的代替任意類型的數據。如下:
- template<class T>
- void show(T x)
- {
- cout<<x<<endl ;
- }
上面的T可以代表任意類型的數據,這樣不是大大減少了函數的重載次數,提高了效率呢。Java是從C++過來的,理解了C++,Java也不在話下。
在Java中自定義范型也可以用在方法上。如下:
1、
//這樣聲明的范型可以代替任意類型數據,我們市場用到的鍵值對Map.Entry<K,V> 不就是給予范型的嗎?
K V都可以代替任意類型的值,但是在Java中范型的實際類型必須是引用類型。
- <K,V> void get(K k,V v)
- {
- }
2、Java中的范型不能像C++那麼靈活。
- <T> T add(T a,T b)
- {
- //return a+b ;//很多人以為Java也想C++一樣可以這樣 ,但是不可以 。
- return null;
- }
這個返回的null 也是有類型限制的,比如上面的a b分別是Integer和String 那麼就會取他們共同的基類Object做為返回值類型,其他的同理。
3、實現任意類型的數組的成員值的交換,注意在自定義范型中范型的實際類型只能是引用數據類型,不能是基本數據類型。
- public static <T> void swap(T[]a,int x,int y)
- {
- T tem =a[x] ;
- a[x]=a[y] ;
- a[y]=tem ;
- }
上面這個方法如果我swap(new Integer[]{1,2,3,4,5},1,2); //這樣就會自動交換下標1 2的值
但是這樣調用就錯了swao(new int[]{1,2,3,5,6},2,3) ; //所以說Java的范型的實際類型只能是引用數據類型
4、
<T extends String> 表示類型只能是String或者String的派生類
<T super String > 表示范型類型只能是String或者String的父類
用法同上
5、下面這個函數利用范型來實現類型自動轉換的功能。
- public static <T> T autoConvert(Object obj)
- //因為返回值是 T標識任意類型 所喲可以 將返回結果賦值給任意類型對象
- {
- return (T)obj;
- }
- Object obj=="";
- String str=autoConvert(obj);
可以完成自動轉換,因為范型T代表任意類型,因此他可以賦值給String類型的對象。
6、將任意類型的對象填充到任意類型的數組中,與是fillArray(new Integer[]{2,3,4},"ddd"); 這樣調用是正確的,這樣做忽略類型限制。
- public static <T> void fillArray(T[] a,T b) //將任意一個對象填充到任意類型的數組
- {
- for(int i =0;i<a.length;i++)
- {
- a[i] =b ;
- }
- }
7、以自定義范型的形式顯示一個集合的數據,下面一個是利用自定義范型一個是利用通配符來實現,但是不同的是利用通配符操作的集合不能向集合中插入元素
但是自定義范型卻可以。原因是通配符代表的集合我們不知道集合內部具體元素 是什麼類型 所以 不能對集合進行add操作。
- public static <T> void showCollection(Collection<T> col,T obj) //利用范型來輸出任意類型集合
- {
- col.add(obj) ;
- for(T a:col)
- {
- System.out.println(a);
- }
- }
- public static void showCollection(Collection<?> col) //利用范型來輸出任意類型集合
- {
- for(Object obj:col)
- {
- System.out.println(obj);
- }
- }
8、如果一個類中多個方法都需要范型,那麼就是用類級別的范型。
例如:
- class A<E>
- {
- public void add(E obj){}
- public E get(){}
- private E data;
- }
這樣聲明范型和在函數前面聲明其實是一樣的,只不過是在類的級別上作用於整個類而已。
9、 要注意范型只是給編譯器看的。
也就是說Vector<Integer> Vector<String> 他們用到的都是同一份字節碼,字節碼只有class文件加載到內存中的時候才有。
所以在一個類中下面2個方法不能同時存在。
void show(Vector<Integer>) {}
void show(Vector<String>){}
這兩個方法都不是重載,因為編譯後要去掉類型信息。