Arraylist Java核心技术卷一 184页

没有泛型类时,原始的ArrayList类提供的get方法别无选择只能返回Obejct,因此,get方法的调用必须对返回值进行类型转换:
Employee e = (Employee)staff. get(i):
原始的ArrayList存在一定的危险性。 它的add和set方法允许接收任意类型的对象。
对于下面这个调用
staff.set(i.new Date());
编译时不会给出任何警告,只有在检索对象并试图对它进行类型转换时,才会发现有问题。如果使用ArrayList,编译器就会检测到这个错误。

第一个问题:Employee e = (Employee)staff. get(i):

猜测:      object obj = (object) staff.get(i)(向上转型)

          Employee e =(Employee) obj(向下转型)

这种向下转型不是很容易会出错了。 比如boss是Employee的一员,但是Employee不一定是就是boss啊。
2.为什么说add和set方法允许接收任意类型的对象的时候,就会出现一定的风险呢?
3. 为什么说只有在检索对象并试图对它进行类型转换时,才会发现问题呢? 如果使用ArrayList,编译器就会检测到错误。

4个回答

你需要知道如下几点:
1、任何类型都继承自object类。
2、原始ArrayList里面存储的是object类型,也就是我可以同时把string、enum、int、double、bool、自定义类等等都可以放到一个集合里面。那么你取出来时,也是object类型,此时如果不做类型检查,那么直接转为Employee类是不安全的,因为可能我存储的是int类型的数字0。所以原始ArrayList是不安全的。
3、如果使用泛型ArrayList,那么在声明的时候就指定了这个ArrayList可以存储的数据类型:如 ArrayList arr;那么这个arr中就只能存储string类型的数据。那么我取出来的时候就不需要检查。因为一定会是一个string数据,这是泛型限定了的,限定这个ArrayList中只能保存你指定的类型数据。

1.在向下转型之前最好做一下判断 是否能转 如果你set的时候是Employee 那么实际上编译器是进行类型擦除也就是说 编译器并不知道你的类型是Employee
2.这个问题就是1中说的类型擦除引发的
3.泛型在使用的时候才会加上类型参数

1.向下转型不是很容易会出错了?
答:是的,所以需要检查类型,用instanceof判断呗
2为什么说add和set方法允许接收任意类型的对象的时候,就会出现一定的风险呢?
答:一个ArrayList如果不加泛型,当你调用add方法时候,不管你添加什么类型的对象都可以,因为默认是Object.
但是你的ArrayList里面对象类型不一致会让你无法处理这些对象,要知道不同类型的对象会有不同的方法,如果你的ArrayList里面有几万个不同类型对象
。你怎么处理?调用get方法获取对象怎么知道它是什么类型?做不到的。。。因此使用泛型限定一个ArrayList只能放一个类型的对象,就很好操作啦。一旦你写了泛型,那么你对这个ArrayList做操作的时候,不能add其它类型的对象。

kbc_kbc
kbc_kbc 为什么不能add其它类型的对象?因为你加了泛型限定,告诉编译器,这个ArrayList只能添加泛型限定的类型
2 年多之前 回复

如果有帮助,望采纳,谢谢!

Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!
立即提问