当前位置:首页 > 编程教程 > java技术文章 > JDK1.6集合框架bug 6260652解析

深入分析JDK1.6集合框架bug 6260652

  • 发布时间:
  • 作者:码农之家
  • 点击:93

这篇文章主要知识点是关于JDK1.6、bug、集合框架、6260652、的内容,如果大家想对相关知识点有系统深入的学习,可以参阅以下电子书

JavaScript核心技术开发解密
  • 类型:JavaScript大小:68.4 MB格式:PDF作者:阳波
立即下载

JDK1.6集合框架bug 6260652解析

最近在看JDK的源码:CopyOnWriteArrayList.java和ArrayList.java,这2个类的构造函数,注释中有一句话看不懂。

public ArrayList(Collection<? extends E> c) { 
elementData = c.toArray(); 
size = elementData.length; 
// c.toArray might (incorrectly) not return Object[] (see 6260652) 
if (elementData.getClass() != Object[].class) 
  elementData = Arrays.copyOf(elementData, size, Object[].class); 
} 
 
public CopyOnWriteArrayList(Collection<? extends E> c) { 
Object[] elements = c.toArray(); 
// c.toArray might (incorrectly) not return Object[] (see 6260652) 
if (elements.getClass() != Object[].class) 
  elements = Arrays.copyOf(elements, elements.length, Object[].class); 
setArray(elements); 
} 

上网查了一下资料,才知道see 6260652 这个编号代表JDK bug库中的编号。可以去官网查看bug详情
http://bugs.java.com/bugdatabase/view_bug.do?bug_id=6260652
http://bugs.java.com/bugdatabase/view_bug.do?bug_id=6515694
6260652 和6515694这2个bug,貌似是同一个问题。这个bug是什么意思呢?我们先来看看一些测试代码:

public static void test1() 
{ 
  SubClass[] subArray = {new SubClass(), new SubClass()}; 
  System.out.println(subArray.getClass()); 
 
  // class [Lcollection.SubClass; 
  BaseClass[] baseArray = subArray; 
  System.out.println(baseArray.getClass()); 
 
  // java.lang.ArrayStoreException 
  baseArray[0] = new BaseClass(); 
} 
 
public static void test2() 
{ 
  List<String> list = Arrays.asList("abc"); 
 
  // class java.util.Arrays$ArrayList 
  System.out.println(list.getClass()); 
 
  // class [Ljava.lang.String; 
  Object[] objArray = list.toArray(); 
  System.out.println(objArray.getClass()); 
 
  objArray[0] = new Object(); // cause ArrayStoreException 
} 
 
public static void test3() 
{ 
  List<String> dataList = new ArrayList<String>(); 
  dataList.add("one"); 
  dataList.add("two"); 
 
  Object[] listToArray = dataList.toArray(); 
 
  // class [Ljava.lang.Object;返回的是Object数组 
  System.out.println(listToArray.getClass()); 
  listToArray[0] = ""; 
  listToArray[0] = 123; 
  listToArray[0] = new Object(); 
   
} 

1、关于test1()

 SubClass 继承自BaseClass,由于SubClass数组中每一个元素都是SubClass对象,所以BaseClass[] baseArray = subArray;这种强制类型转换不会报错。这其实就是java对象的向上转型,子类数组转换成父类数组是允许的。但是由于数组中元素类型都是SubClass类型的,所以 baseArray[0] = new BaseClass();会报错java.lang.ArrayStoreException。这也就是说假如我们有1个Object[]数组,并不代表着我们可以将Object对象存进去,这取决于数组中元素实际的类型。

2、关于test2()

List<String> list = Arrays.asList("abc");需要注意,可以知道返回的实际类型是java.util.Arrays$ArrayList,而不是ArrayList。我们调用

Object[] objArray = list.toArray();返回是String[]数组,所以我们不能将Object对象,放到objArray数组中。

3、关于test3()

ArrayList对象的toArray()返回就是Object[]数组,所以我们可以将任意对象存放到返回的Object[]数组中。

通过test2和test3可以看出,如果我们有1个List<String> stringList对象,当我么调用Object[] objectArray = stringList.toArray();的时候,objectArray 并不一定能够放置Object对象。这就是源码中的注释:c.toArray might (incorrectly) not return Object[] (see 6260652)。为了考虑这种情况,所以源码中进行了if判断,来防止错误的数组对象导致异常。Arrays.copyOf(elementData, size, Object[].class);这个方法就是用来创建1个Object[]数组,这样数组中就可以存放任意对象了。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持码农之家。

以上就是本次给大家分享的关于java的全部知识点内容总结,大家还可以在下方相关文章里找到相关文章进一步学习,感谢大家的阅读和支持。

Java 相关电子书
学习笔记
网友NO.104968

java集合框架线程同步代码详解

List接口的大小可变数组的实现。实现了所有可选列表操作,并允许包括null在内的所有元素。除了实现List接口外,此类还提供一些方法来操作内部用来存储列表的数组的大小。(此类大致上等同于Vector类,除了此类是不同步的。)size、isEmpty、get、set、iterator和listIterator操作都以固定时间运行。add操作以分摊的固定时间运行,也就是说,添加n个元素需要O(n)时间。其他所有操作都以线性时间运行(大体上讲)。与用于LinkedList实现的常数因子相比,此实现的常数因子较低。每个ArrayList实例都有一个容量。该容量是指用来存储列表元素的数组的大小。它总是至少等于列表的大小。随着向ArrayList中不断添加元素,其容量也自动增长。并未指定增长策略的细节,因为这不只是添加元素会带来分摊固定时间开销那样简单。在添加大量元素前,应用程序可以使用……

网友NO.383698

JAVA集合框架工具类自定义Collections集合方法

项目中有需要多次统计 某些集合中 的某个属性值,所以考虑封装一个方法,让其其定义实现计算方式。 话不多说,看代码: 1、封装的自定义集合工具类:CollectionsCustom package com.test.util;import java.util.Collection;import org.apache.commons.collections.CollectionUtils;/** * 自定义集合处理类 */public class CollectionsCustom { /** * 将传入的collection内对象进行计算后得出结果 * @param original 计算前collection * @param reduceFunction 计算方式 * @param initValue 计算结果初始值 * @param Input collection对象类型 * @param Output 结果类型 * @return */ public static Input, Output Output reduce(CollectionInput original, Output initValue, ReduceFunctionInput, Output reduceFunction) { Output result = initValue; if (CollectionUtils.isEmpty(original)) { return result; } if (reduceFunction == null) { return result; } for (Input input : original) { result = reduceFunction.apply(input, result); } re……

网友NO.900455

Java复习之集合框架总结

俗话说:温故而知新。想想学过的知识,就算是以前学得很不错,久不用了,就会忘记,所以温习一下以前学习的知识我认为是非常有必要的。而本篇文件温习的是 Java基础中的集合框架。 为什么会有集合框架? 平时我们用数组存储一些基本的数据类型,或者是引用数据类型,但是数组的长度是固定的,当添加的元素超过了数组的长度时,需要对数组进行重新的定义,这样就会显得写程序太麻烦,所以Java内部为了我们方便,就提供了集合类,能存储任意对象,长度是可以改变的,随着元素的增加而增加,随着元素的减少而减少。 数组可以存储基本数据类型,也可以存储引用数据类型,基本数据类型存储的是值,引用数据类型存储的是地址,而集合只能存储引用数据类型(也就是对象),其实集合中也可以存储基本数据类型,但是在存储的时候会自……

网友NO.434818

Java集合框架源码分析之LinkedHashMap详解

LinkedHashMap简介 LinkedHashMap是HashMap的子类,与HashMap有着同样的存储结构,但它加入了一个双向链表的头结点,将所有put到LinkedHashmap的节点一一串成了一个双向循环链表,因此它保留了节点插入的顺序,可以使节点的输出顺序与输入顺序相同。 LinkedHashMap可以用来实现LRU算法(这会在下面的源码中进行分析)。 LinkedHashMap同样是非线程安全的,只在单线程环境下使用。 LinkedHashMap源码剖析 LinkedHashMap源码如下(加入了详细的注释): package java.util; import java.io.*; public class LinkedHashMapK,V extends HashMapK,V implements MapK,V { private static final long serialVersionUID = 3801124242820219131L; //双向循环链表的头结点,整个LinkedHashMap中只有一个header, //它将哈希表中所有的Entry贯穿起来,header中不保存key-value对,只保存前后节点的引用 private transient EntryK,V header; //双向链表中元素排序规……

<
1
>

Copyright 2018-2020 www.xz577.com 码农之家

版权投诉 / 书籍推广 / 赞助:520161757@qq.com