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