分享
 
 
 

ArrayList使用toArray()构造数组时的问题

王朝java/jsp·作者佚名  2006-01-10
窄屏简体版  字體: |||超大  

标题:ArrayList使用toArray()构造数组时的问题

关键字:toArray 构造数组

作者:jrq

摘要:解决使用ArrayList.toArray()构造数组时的问题。做备忘。

本文链接: http://blog.csdn.net/jrq/archive/2005/10/27/517428.aspx

正文:

1. 为了方面按列作外循环,想把ArrayList构造成一个二维数组,如下:

......

ArrayList result=GetResult();

int n=result.size();

String[][] myArray=new String[n][]; //定义二维数组

for (int i=0;i<n;i++) //构造二维数组

{

ArrayList tempArray= (ArrayList)result.get(i);

myArray[i]=(String[])tempArray.toArray();

}

......

程序可以编译通过。

但在运行到myArray[i]=(String[])tempArray.toArray()时,出现java.lang.ClassCastException的错误,很奇怪。

花了一晚上时间,查了N多资料,总算搞定了。现把问题记录下来,以备参考。

2. 此事从头说起。

ArrayList类扩展AbstractList并执行List接口。ArrayList支持可随需要而增长的动态数组。

ArrayList有如下的构造函数:

ArrayList( )

ArrayList(Collection c)

ArrayList(int capacity)

如果调用new ArrayList()构造时,其默认的capacity(初始容量)为10。

参见ArrayList源码,其中是这样定义的:

public ArrayList() {

this(10);

}

默认初始化内部数组大小为10。为什么是10?不知道。可能SUN觉得这样比较爽吧。

程序编译后执行ArrayList.toArray(),把ArrayList转化为数组时,该数组大小仍为capacity(为10)。

当装入的数据和capacity值不等时(小于capacity),比如只装入了5个数据,数组中后面的(capacity - size)个对象将置为null,此时当数组强制类型转换时,容易出现一些问题,如java.lang.ClassCastException异常等。

解决办法是:在用ArrayList转化为数组装数据后,使用trimToSize()重新设置数组的真实大小。

3. 本例修改后的代码修如下,可顺利运行:

for (int i=0;i<n;i++) //构造二维数组

{

ArrayList tempArray= (ArrayList)result.get(i);

myArray[i]=(String[])tempArray.toArray(new String[0]); //注意此处的写法

}

看看下面这些也许就明白了--

ArrayList.toArray()之一:

public Object[] toArray() {

Object[] result = new Object[size];

System.arraycopy(elementData, 0, result, 0, size);

return result;

}

返回ArrayList元素的一个数组,注意这里虽然生成了一个新的数组,但是数组元素和集合中的元素是共享的,Collection接口中说这个是安全的,这是不严格的。

下面的例子演示了这个效果。

ArrayList al=new ArrayList();

al.add(new StringBuffer("hello"));

Object[] a=al.toArray();

StringBuffer sb=(StringBuffer)a[0];

sb.append("changed"); //改变数组元素同样也改变了原来的ArrayList中的元素

System.out.println(al.get(0));

这里不要用String来代替StringBuffer,因为String是常量。

ArrayList.toArray()之二:

public Object[] toArray(Object a[]) {

if (a.length < size)

a = (Object[])java.lang.reflect.Array.newInstance(a.getClass().getComponentType(), size);

System.arraycopy(elementData, 0, a, 0, size);

if (a.length > size)

a[size] = null;

return a;

}

这个方法有可能不需要生成新的数组,注意到如果数组a容量过大,只在size处设置为null。

如果这个数组a足够大,就会把数据全放进去,返回的数组也是指向这个数组,(数组多余的空间存储的是null对象);要是不够大,就申请一个跟参数同样类型的数组,把值放进去,然后返回。

4. 网上的资料一:

public String[] getPlatformIDList()

{

Vector result = new Vector();

try

{

Statement stmt = conn.createStatement();

String sql = "SELECT PlatformID FROM Platform";

rs = stmt.executeQuery(sql);

while(rs.next())

{

result.add(rs.getString(1));

}

if (result.size() > 0)

{

String[] str = (String[]) result.toArray(); // 出现ClassCastException

return str;

}

else

return null;

}

catch(Exception e)

{

System.err.println(e);

return null;

}

finally

{

try

{

rs.close();

conn.close();

}

catch(Exception e2)

{}

}

}

程序运行后,发现不能将Vector类经过toArray()方法得到的Object[]直接转换成String[]。

找到用另一个带有参数的 toArray(T[] a)方法才可以。

将该语句改为:

String[] str = (String[]) result.toArray(new String[1]);即告诉Vector,我要得到的数组的类型。

回想一下,应该是java中的强制类型转换只是针对单个对象的,想要偷懒,将整个数组转换成另外一种类型的数组是不行的。

5. 网上的资料二:

正确使用List.toArray()--

在程序中,往往得到一个List,程序要求对应赋值给一个array,可以这样写程序:

Long [] l = new Long[list.size()];

for(int i=0;i<list.size();i++)

l[i] = (Long) list.get(i);

要写这些code,似乎比较繁琐,其实List提供了toArray()的方法。

但是要使用不好,就会有ClassCastExceptiony异常。究竟这个是如何产生的,且看代码:

List list = new ArrayList();

list.add(new Long(1));list.add(new Long(2));

list.add(new Long(3));list.add(new Long(4));

Long[] l = (Long[])list.toArray();

for(int i=0; i<l.length; i++)

System.out.println(l[i].longValue());

红色代码会抛java.lang.ClassCastException。

当然,为了读出值来,你可以这样code:

Object [] a = list.toArray();

for(int i=0;i<a.length;i++)

System.out.println(((Long)a[i]).longValue());

但是让数组丢失了类型信息,这个不是我们想要得。

toArray()正确使用方式如下:

1) Long[] l = new Long[<total size>];

list.toArray(l);

2) Long[] l = (Long []) list.toArray(new Long[0]);

3) Long [] a = new Long[<total size>];

Long [] l = (Long []) list.toArray(a);

6. 总结补充:

java sdk doc 上讲:

public Object[] toArray(Object[] a)

a--the array into which the elements of this list are to be stored, if it is big enough; otherwise, a new array of the same runtime type is allocated for this purpose.

如果这个数组a足够大,就会把数据全放进去,返回的数组也是指向这个数组,(数组多余的空间存储的是null对象);要是不够大,就申请一个跟参数同样类型的数组,把值放进去,然后返回。

需要注意的是:你要是传入的参数为9个大小,而list里面有5个object,那么其他的四个很可能是null , 使用的时候要特别注意。

6. 完毕。

J.R.Q.

2005.10.27凌晨于广州

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有