详解泛型与类型转换

Java 5.0引入泛型以后,提高了类型安全性,然而也给编程带来了新的挑战以及新的误区,类型转换就是一个很好的例子。
首页 新闻资讯 行业资讯 详解泛型与类型转换

Java 5.0引入泛型以后,提高了类型安全性,然而也给编程带来了新的挑战以及新的误区,类型转换就是一个很好的例子。引入泛型以后,对象的类型似乎由两个类型构成:集合类型和参数类型,例如List list = new LinkedList(),似乎list的类型由List和Integer共同决定。如果真是这样,那么我们如果希望对list进行类型转换,我们该怎么办?是对List转换,还是对Integer转换?要解答上面的问题很简单,编写如下的代码,运行之就可看出端倪:

 

复制

public   static   void  wrongConversion()   {          List < Integer >  list  =   new  LinkedList < Integer > ();          list.add( 1 );          list.add( 2 );          list.add( 3 );                     // 下面的代码无法编译通过。           // List wrongList = list;                  }              public   static   void  rightConversion()   {          LinkedList < Integer >  list  =   new  LinkedList < Integer > ();          list.add( 10 );          list.add( 20 );          list.add( 30 );                    List < Integer >  rightList  =  list;           for ( int  number: rightList)   {              System.out.println(number);          }                 }
  • 1.

  • 2.

  • 3.

  • 4.

  • 5.

  • 6.

  • 7.

  • 8.

  • 9.

  • 10.

  • 11.

  • 12.

  • 13.

  • 14.

  • 15.

  • 16.

  • 17.

  • 18.

  • 19.

  • 20.

  • 21.

  • 22.

  • 23.

 

如果你运行上面的代码,你会发现前一个函数根本无法通过编译,而后一个函数不仅编译成功,而且能得到正确结果。这似乎给出了一个结论:对象的类型由集合类型来决定,对对象进行类型转换时也是对集合类型进行转换。事实确实如此!那么为什么会如此呢?

我们可以回忆一下以前我们对参数类型的描述:参数类型不过是一个占位符而已,在编译时,它会被具体的类型代替,同时对添加到集合对象中的数据进行参数检查;如果添加到对象中的数据类型不是指定的具体类型或其子类型,则编译出错;而且,编译以后,该占位符会被移除,运行时你是无法得到任何集合类型中数据的类型信息的。简而言之:

List strings = new LinkedList( );

List ints = new LinkedList( );

上面的代码在运行时和

List strings = new LinkedList( );

List ints = new LinkedList( );

毫无二致。了解了这一点以后,相信我们对于类型转换也会有更深的理解了。

泛型给我们带来了好处,使我们在编译时就能发现很多错误;然而任何事物都是柄双刃剑,泛型也不例外。泛型的缺点最明显之处就在于类型转换。如果你读过前面的文章,肯定看到过不带参数类型的集合对象转换为带参数类型的集合对象,其实不仅仅如此,这两者之间可以任意转换。这就意味着一点,我们不仅可以对集合类型进行转换,也可以对参数类型进行任意的转换。这似乎与前面的描述不符,因为上面的代码中的List wrongList = list;根本无法编译通过。确实如此,这行代码确实无法编译通过,不过我们中间做一步处理以后,上面的转换就可以正常编译了:

List mediaList = list;

List wrongList = mediaList;

由此可见,泛型在给我们带来好处的同时,也带来了无数陷阱,我们在编程时需十分注意,而Java的泛型机制也需要不断更新。

最后和以往一样,以一个拙劣的程序结束本文:

上面程序的输出结果为:

 

复制

rightConversion  10 20 30 badConversion  1 2 3 java.lang.ClassCastException: java.lang.String      at com.jiang.tiger.chap1.GenericReflection.fillList(GenericReflection.java:11)      at com.jiang.tiger.chap1.GenericReflection.badConversion(GenericReflection.java:32)      at com.jiang.tiger.chap1.GenericReflection.main(GenericReflection.java:109)  badConversion2  1 2 java.lang.ClassCastException: java.lang.String      at com.jiang.tiger.chap1.GenericReflection.badConversion2(GenericReflection.java:45)      at com.jiang.tiger.chap1.GenericReflection.main(GenericReflection.java:116)  wrongConversion  100 200 300 java.lang.ClassCastException: java.lang.Double      at com.jiang.tiger.chap1.GenericReflection.wrongConversion(GenericReflection.java:62)      at com.jiang.tiger.chap1.GenericReflection.main(GenericReflection.java:123)  badReflection  Filling list and printing in normal way  1 2 3 1 2 3 4 5 6 java.lang.ClassCastException: java.lang.String      at com.jiang.tiger.chap1.GenericReflection.printList(GenericReflection.java:17)      at com.jiang.tiger.chap1.GenericReflection.badReflection(GenericReflection.java:99)Printing with illegal values in list  1 2 3 1 2 3 4 5 6      at com.jiang.tiger.chap1.GenericReflection.main(GenericReflection.java:130)
  • 1.

  • 2.

  • 3.

  • 4.

  • 5.

  • 6.

  • 7.

  • 8.

  • 9.

  • 10.

  • 11.

  • 12.

  • 13.

  • 14.

  • 15.

  • 16.

  • 17.

  • 18.

  • 19.

  • 20.

  • 21.

  • 22.

  • 23.

  • 24.

  • 25.

  • 26.

  • 27.

  • 28.

  • 29.

  • 30.

  • 31.

  • 32.

  • 33.

  • 34.

  • 35.

  • 36.

  • 37.

  • 38.

  • 39.

  • 40.

  • 41.

  • 42.

  • 43.

  • 44.

  • 45.

  • 46.

  • 47.

  • 48.

  • 49.

  • 50.

【编辑推荐】

Java泛型的理解与等价实现Java泛型编程快速入门 浅谈关于C#、Java泛型的看法 在Java中定义自己的工具库Java技术在协同软件中的应用

22    2011-04-13 09:16:55    泛型