详解泛型与类型转换
创始人
2024-07-25 18:51:10
0

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

 

  1. public   static   void  wrongConversion()   {  
  2.         List < Integer >  list  =   new  LinkedList < Integer > ();  
  3.         list.add( 1 );  
  4.         list.add( 2 );  
  5.         list.add( 3 );  
  6.           
  7.          // 下面的代码无法编译通过。  
  8.          // List wrongList = list;   
  9.            
  10.     }   
  11.       
  12.      public   static   void  rightConversion()   {  
  13.         LinkedList < Integer >  list  =   new  LinkedList < Integer > ();  
  14.         list.add( 10 );  
  15.         list.add( 20 );  
  16.         list.add( 30 );  
  17.           
  18.         List < Integer >  rightList  =  list;  
  19.          for ( int  number: rightList)   {  
  20.             System.out.println(number);  
  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的泛型机制也需要不断更新。

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

上面程序的输出结果为:

 

  1. rightConversion  
  2. 10 
  3. 20 
  4. 30 
  5. badConversion  
  6. java.lang.ClassCastException: java.lang.String  
  7.     at com.jiang.tiger.chap1.GenericReflection.fillList(GenericReflection.java:11)  
  8.     at com.jiang.tiger.chap1.GenericReflection.badConversion(GenericReflection.java:32)  
  9.     at com.jiang.tiger.chap1.GenericReflection.main(GenericReflection.java:109)  
  10. badConversion2  
  11. java.lang.ClassCastException: java.lang.String  
  12.     at com.jiang.tiger.chap1.GenericReflection.badConversion2(GenericReflection.java:45)  
  13.     at com.jiang.tiger.chap1.GenericReflection.main(GenericReflection.java:116)  
  14. wrongConversion  
  15. 100 
  16. 200 
  17. 300 
  18. java.lang.ClassCastException: java.lang.Double  
  19.     at com.jiang.tiger.chap1.GenericReflection.wrongConversion(GenericReflection.java:62)  
  20.     at com.jiang.tiger.chap1.GenericReflection.main(GenericReflection.java:123)  
  21. badReflection  
  22. Filling list and printing in normal way  
  23. java.lang.ClassCastException: java.lang.String  
  24.     at com.jiang.tiger.chap1.GenericReflection.printList(GenericReflection.java:17)  
  25.     at com.jiang.tiger.chap1.GenericReflection.badReflection(GenericReflection.java:99)Printing with illegal values in list  
  26.  
  27.     at com.jiang.tiger.chap1.GenericReflection.main(GenericReflection.java:130)  

【编辑推荐】

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

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
Windows恶意软件20年“... 在Windows的早期年代,病毒游走于系统之间,偶尔删除文件(但被删除的文件几乎都是可恢复的),并弹...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...
《非诚勿扰》红人闫凤娇被曝厕所... 【51CTO.com 综合消息360安全专家提醒说,“闫凤娇”、“非诚勿扰”已经被黑客盯上成为了“木...