Object[]数组的转换问题

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

......

ArrayList result=GetResult();

int n=result.size();

String[][] myArray=new String[n][];//定义二维数组
 
  for (int i=0;i<n;i++) //构造二维数组
  {
    ArrayListtempArray= (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)

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

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

publicArrayList() {
  this(10);
    }

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

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

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

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

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

for (inti=0;i<n;i++)  //构造二维数组
     {
          ArrayList tempArray=(ArrayList)result.get(i);
          myArray[i]=(String[])tempArray.toArray(newString[0]);  //注意此处的写法
      }

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

ArrayList.toArray()之一:

public Object[] toArray() {
  Object[] result = new Object[size];
  System.arraycopy(elementData, 0, result, 0, size);
  return result;
}

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

下面的例子演示了这个效果。
 
   ArrayList al=newArrayList();
   al.add(newStringBuffer("hello"));
   Object[] a=al.toArray();
   StringBuffersb=(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(newString[1]);即告诉Vector,我要得到的数组的类型。

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

5. 网上的资料二:

正确使用List.toArray()--
 
  在程序中,往往得到一个List,程序要求对应赋值给一个array,可以这样写程序:
 
    Long [] l =new Long[list.size()];
    for(inti=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[<totalsize>];
            list.toArray(l);
 
       2)  Long[] l = (Long []) list.toArray(newLong[0]);

3)  Long [] a = newLong[<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 thesame  runtime type is allocated for thispurpose.

如果这个数组a足够大,就会把数据全放进去,返回的数组也是指向这个数组,(数组多余的空间存储的是null对象);要是不够大,就申请一个跟参数同样类型的数组,把值放进去,然后返回。
 
    需要注意的是:你要是传入的参数为9个大小,而list里面有5个object,那么其他的四个很可能是null, 使用的时候要特别注意。

7. 完毕。

时间: 2024-10-31 10:16:37

Object[]数组的转换问题的相关文章

C#结构体和字节数组的转换函数

在通信过程中,一般我们都会操作到字节数组.特别是希望在不同语言编程进行操作的时候. 虽然C#提供了序列化的支持,不用字节数组也行.但操作字节数组肯定会碰到. 一般都会采用结构来表示字节数组.但结构与字节数组直接的转换实在很麻烦. 字节操作不但容易出错,而且每增加一个结构,就自己实现一遍,实在是烦不胜烦. 有没有简单的方法呢?当然有.可以采用非托管区的一些方法来实现. 首先,导入命名空间:System.Runtime.InteropServices; 定义结构的时候,要给结构指定特性. 如: //

Java读取json数组转化成List或Object数组

在之前的开发中经常遇到将List或Array转换成json传递到web前端, 供前端显示,但是今天我遇到了一个需要将json数组传递到后台,并在后 台转换成list的问题.为此我花费了一段较长的时间,为此我写下这篇博客. 首先是在前端用js构造json数组,html代码如下: <TABLE style="width: 900px" id="table"> <pre name="code" class="html&quo

给object数组进行排序(排序条件是每个元素对象的属性个数)

从汤姆大叔的博客里看到了6个基础题目:本篇是第3题 - 给object数组进行排序(排序条件是每个元素对象的属性个数) 解题关键: 1.Array.sort的用法 2.object的属性数量的统计 解点1:Array.sort的用法 Array.sort可以为数组指定一个排序规则,一般用如下格式进行指定,代码如下: var arr = [10,6,0,4]; console.log( arr.sort() ); //按字符排序 0,10,4,6 console.log( arr.sort( fu

JAXB的XML和Object的自动转换

JAXB可以实现XML和Object的自动转换,这样我们就不用再自己写xml的解析了. 1.从xml文件读取自动转换成Object 2.从xml字符串读取自动转换成Object 3.把对象转化成xml输出到控制台 4.把对象转化成xml输出到文件 5.把对象转化成xml字符串 vdi.xml <?xml version="1.0" encoding="UTF-8"?> <vdis size="2" batch_number=&q

Java反射获取实体的所有可见属性值,返回Object数组

获取实体的所有可见属性值 以下代码提供了两种实现,一种是基于List,一种是基于Map; 基于List的实现更节省内存,更高效一些:如果你有其它特殊的需求,可以根据实际参考以下代码进行扩展,或许有需要用到Map的情况呢! 当然,使用BeanUtils.describe或PropertyUtils.describe一两句代码就搞定了,但还需要额外添加包引用:另外效率方面是不是更高呢?我没有做过测试. /** * 获取实体的所有可见属性值 * @param object 实体类的实例 * @retu

图片和base64编码字符串 互相转换,图片和byte数组互相转换

图片和base64编码字符串 互相转换 import sun.misc.BASE64Decoder; import sun.misc.BASE64Encoder; import java.io.*; /** * @author lishupeng * @create 2017-05-06 下午 2:56 **/ public class Base64Test { public static void main(String[] args) { String strImg = GetImageSt

【ThinkingInJava】5、以Object数组作为参数

/** * 书本:<Thinking In Java> * 功能:以Object数组作为参数 * 文件:VarArgs.java * 时间:2014年10月6日20:04:18 * 作者:cutter_point */ package Lesson5InitializationAndCleanUp; class A{} public class VarArgs { static void printArray(Object [] args) { for(Object obj : args) /

Java 复习 —— 集合与数组的转换

前言: 很多时候,你会觉得数组使用方便,有些时候你会觉得集合使用更加方便,你们对于集合和数组的转换那就在所难免了,下面总结一下数组与集合的转换. 1.把数组转为集合 Arrays.asList()  1)网上解释 在使用Arrays.asList()后调用add,remove这些method时出现 java.lang.UnsupportedOperationException 异常.这是由于Arrays.asList() 返回java.util.Arrays$ArrayList, 而不是Arra

16进制值字符串与byte数组的转换

/** * 将byte数组转换为表示16进制值的字符串, 如:byte[]{8,18}转换为:0813, 和public static byte[] * hexStr2ByteArr(String strIn) 互为可逆的转换过程 * * @param arrB * 需要转换的byte数组 * @return 转换后的字符串 * @throws Exception * 本方法不处理任何异常,所有异常全部抛出 */ public static String byteArr2HexStr(byte[