java 泛型的生命周期_Jdk1.5中的新特性 --泛型 (详细版)

本来只转载了个链接,和一个简单的使用程序,但昨天不小心看到有人批判jdk1.5,先说java要强制转型不好的问题没解决,

容器不能放基类型不好,接着说泛型没用。而恰恰Jdk1.5中解决了这些问题,所以感叹之余,把这篇文章改一下,详细的说说泛型。

一,Java中的泛型:

在Java中能使用到泛型的多是容器类,如各种list map set,因为Java是单根继承,所以容器里边可以放的

内容是任何Object,所以从意义上讲原本的设计才是泛型。但用过Java的人是否感觉每次转型很麻烦呢?

而且会有些错误,比如一个容器内放入了异质对象,强制转型的时候会出现cast异常。而这中错误在编译器是

无从发现的。所以jdk1.5中提供了泛型,这个泛型其实是向c++靠拢了.好,我们先看几个实例再细说原理。

二,泛型的用法:(多个实例)

1

1070342_1.gif 实例A2 1070342_1.gifArrayList < String > strList = new ArrayList < String > ();3 1070342_1.gifstrList.add( " 1 " );4 1070342_1.gifstrList.add( " 2 " );5 1070342_1.gifstrList.add( " 3 " );6 1070342_1.gif// 关键点(1) 注意下边这行,没有强制转型 7 1070342_1.gifString str = strList.get( 1 );8 1070342_1.gif// 关键点(2)然後我们加入,这个时候你会发现编译器报错,错误在编译器被发现,错误当然是发现的越早越好 9 1070342_1.gifstrList.add( new Object());

1

1070342_1.gif 实例B2 1070342_1.gifArrayList < Integer > iList = new ArrayList < Integer > ();3 1070342_1.gif// 关键点(3) 注意直接把整数放入了集合中,而没有用Integer包裹 4 1070342_1.gifiList.add( 1 );5 1070342_1.gifiList.add( 2 );6 1070342_1.gifiList.add( 3 );7 1070342_1.gif// 关键点(4)同样直接取出就是int 8 1070342_1.gifint num = iList.get( 1 );

1

1070342_1.gif 实例C2 1070342_1.gif// 关键点(5)展示一下key-value的时候要怎么写,同时key和value也可以是基本类型了。 3 1070342_1.gifHashMap < Integer,Integer > map = new HashMap < Integer,Integer > ();4 1070342_1.gifmap.put( 1 , 11 );5 1070342_1.gifmap.put( 2 , 22 );6 1070342_1.gifmap.put( 3 , 33 );7 1070342_1.gifint inum = map.get( 1 );8 1070342_1.gif

三,看完了实例了,详细来说说为什么吧

首先jdk1.5中的泛型,第一个解决的问题,就是Java中很多不必要的强制转型了,具体的实现,我们以ArrayList

为例,下边是ArrayList中的片断代码:

1

1070342_1.gifArrayList类的定义,这里加入了21070342_1.gifpublicclassArrayListextendsAbstractList31070342_1.gifimplementsList, RandomAccess, Cloneable, java.io.Serializable41070342_1.gif51070342_1.gif//get方法,返回不再是Object 而是E61070342_31.gif

1070342_32.gifpublicE get(intindex)1070342_33.gif{71070342_34.gif    RangeCheck(index);81070342_34.gifreturnelementData[index];91070342_36.gif}101070342_1.gif//add方法,参数不再是Object 而是E111070342_31.gif

1070342_32.gifpublicbooleanadd(E o)1070342_33.gif{121070342_34.gif    ensureCapacity(size+1);//Increments modCount!!131070342_34.gifelementData[size++]=o;141070342_34.gifreturntrue;151070342_36.gif}161070342_1.gif

四,Boxing 和UnBoxing

看到上边的关键点(3)和(4)是否感觉惊奇呢,因为Java中烦人的除了强制转型,另一个就是基础类型了

放入容器的时候要包装,取出了还要转回。Jdk1.5中解决了这个问题.如上边的使用方法

五,泛型的生命周期(使用注意事项)

如果我们试着把ArrayList list的内容序列化,然後再读取出来,在使用的过程中会发现出错,

为什么呢?用Stream读取一下回来的数据,你会发现不见了,list变成了普通的ArrayList,而不是

参数化型别的ArrayList了,为什么会这样呢 ?见下边的比较

六,C++的泛型和Java的泛型

在泛型的实现上,C++和Java有着很大的不同,

Java是擦拭法实现的

C++是膨胀法实现的

因为Java原本实现就是泛型的,现在加入型别,其实是"窄化",所以采用擦拭法,在实现上,其实是封装了原本的

ArrayList,这样的话,对于下边这些情况,Java的实现类只有一个。

1

1070342_1.gifArrayList1070342_33.gif.;publicclassArrayList21070342_1.gifArrayList1070342_33.gif..;--同上--31070342_1.gifArrayList1070342_33.gif..;--同上--41070342_1.gif而C++采用的是膨胀法,对于上边的三种情况实际是每一种型别都对应一个实现,实现类有多个51070342_1.giflistli;classlist;//int 版本61070342_1.giflistls;classlist;//string 版本71070342_1.giflistld;classlist;//double 版本

这就造成了,在序列化后,Java不能分清楚原来的ArrayList是

ArrayList还是ArrayList

七,题外话,在很多东西的实现上C++和Java有很多不同

例如运算符的问题i=i++问题,详细看这里

例如在C++中能很好实现的double-checked locking单态模式,在Java中几乎很难实现 详细看这里

还有就是上边提到的泛型实现上。

八,Jdk 1.5加入了不少新东西,有些能很大的提高开发质量,例如Jdk1.4 ,Jdk.15中StringBuffer的不同

因为从1。4转入1。5不久,所以慢慢会发一些在1。5的使用过程中发现的东西。

最后,我们还可以自己写类似ArrayList这样的泛型类,至于如何自定义泛型类,泛型方法请参见候捷先生的文章

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值