qileilove

blog已经转移至github,大家请访问 http://qaseven.github.io/

Java自定义范型的应用技巧

 我们在JAVA中处处都用到了范型,JAVA中的范型是从C++模板继承来的,不过JAVA的范型的功能远远没有C++那么强大。

  我们知道在C++中模板可以很方便的代替任意类型的数据如下;

template<class T>
void show(T x)
{
cout<<x<<endl ;
}

  上面的T可以代表任意类型的数据,这样不是大大减少了函数的重载次数,提高了效率呢。java是从C++过来的,理解了C++,jav也不在话下。

  在java中自定义范型也可以用在方法上如下:

  1、//这样声明的范型可以代替任意类型数据我们市场用到的键值对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、实现任意类型的数组的成员值的交换,注意在自定义范型中范型的实际类型只能是引用数据类型不能是基本数据类型

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的值

  但是这样调用就错了   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(newInteger[]{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>){}

  这两个方法都不是重载因为编译后要去掉类型信息。

posted on 2012-02-21 16:58 顺其自然EVO 阅读(211) 评论(0)  编辑  收藏


只有注册用户登录后才能发表评论。


网站导航:
 
<2012年2月>
2930311234
567891011
12131415161718
19202122232425
26272829123
45678910

导航

统计

常用链接

留言簿(55)

随笔分类

随笔档案

文章分类

文章档案

搜索

最新评论

阅读排行榜

评论排行榜