`
uule
  • 浏览: 6358202 次
  • 性别: Icon_minigender_1
  • 来自: 一片神奇的土地
社区版块
存档分类
最新评论

java.lang.reflect.Array.newInstance与list.toArray

 
阅读更多

 

一.

import java.util.ArrayList;  
import java.util.List;  
public class Test {  
    public static void main(String[] args) {  
          
        List<String> list = new ArrayList<String>();   
        list.add("1");   
        list.add("2");   
        String[] tt =(String[]) list.toArray(new String[0]);   
    }  
}

  

这段代码是没问题的,但我们看到String[] tt =(String[]) list.toArray(new String[0]) 中的参数很奇怪,然而去掉这个参 数new String[0]却在运行时报错。。。

 

二.经研究发现toArray有两个方法:

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

 

不带参数的toArray方法,是构造的一个Object数组, 然后进行数据拷贝,此时进行转型就会产生ClassCastException,这也就是上述问题的root cause了。

  

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;  
}

 

而带参数的toArray方法,则是根据参数数组的类型,构造了一个对应类型的,长度跟ArrayList的size一致的空数组,虽然方法本身还是以 Object数组的形式返回结果,不过由于构造数组使用的ComponentType跟需要转型的ComponentType一致,就不会产生转型异常。

 

①不带参数的toArray方法,是构造的一个Object数组,然后进行数据拷贝,此时进行转型就会产生ClassCastException 

②带参数的toArray方法,则是根据参数数组的类型,构造了一个对应类型的,长度跟ArrayList的size一致的空数组,虽然方法本身还是以 Object数组的形式返回结果,不过由于构造数组使用的ComponentType跟需要转型的ComponentType一致,就不会产生转型异常

 

三.因此在使用toArray的时候可以参考以下方式

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

 

2. Long[] a = new Long[<total size>];

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

 

四.该容器中的元素已经用泛型限制了,那里面的元素就应该被当作泛型类型的来看了,然而在目前的java中却不是的,当直接String[] tt =(String[]) list.toArray()时,运行报错。回想一下,应该是java中的强制类型转换只是针对单个对象的,想要偷懒,将整个数组转换成另外一种类型的数 组是不行的,这和数组初始化时需要一个个来也是类似的。

 

 java.lang.reflect.Array.newInstance

 

1、java.lang.reflect.Array 类提供了动态创建和访问数组元素的各种静态方法

如下例子创建了一个长度为10 的字符串数组,接着把索引位置为5 的元素设为“hello”,然后再读取索引位置为5 的元素的值

import java.lang.reflect.Array;   
  
public class ArrayTest {   
  
    public static void main(String[] args) throws Exception {   
           
        Class clazz = Class.forName("java.lang.String");   
           
        //创建一个长度为10的字符串数组,在Java中数组也可以作为Object对象   
        Object array = Array.newInstance(clazz, 10);   
           
        //把字符串数组对象的索引位置为5的元素设置为"hello"   
        Array.set(array, 5, "hello");   
           
        //获得字符串数组对象的索引位置为5的元素的值   
        String str = (String)Array.get(array, 5);   
        System.out.println(str);//hello   
    }  
  
}  

 

2、Java中动态修改数组(Array)的大小

 

private static Object resizeArray (Object oldArray, int newSize) {

	int oldSize = java.lang.reflect.Array.getLength(oldArray);
	
	Class elementType = oldArray.getClass().getComponentType();
	
	Object newArray = java.lang.reflect.Array.newInstance(
			elementType,newSize);
		
	int preserveLength = Math.min(oldSize,newSize);
	if (preserveLength > 0)
		System.arraycopy (oldArray,0,newArray,0,preserveLength);
		
	return newArray; 
}

 int oldSize = java.lang.reflect.Array.getLength(oldArray); //获取数组oldArray的长度 

 

Class elementType = oldArray.getClass().getComponentType(); //获取数组oldArray的元素类型 是字符串 整数 还是其他类的。得到的结果是一个class

 

Object newArray = java.lang.reflect.Array.newInstance(elementType,newSize); //实例一个新的数组 类型和oldArray的一样 长度参数传入的newSize

 

int preserveLength = Math.min(oldSize,newSize);   //得到新数组newArray 和oldArray两个中长度最短的,并把长度返回给preserveLength

之所以这么实现是因为考虑到复制所有类型的数组,所以把数组向上转型成Object了。

 

 

public static void main (String[] args) {
   int[] a = {1,2,3};
   a = (int[])resizeArray(a,5);
   a[3] = 4;
   a[4] = 5;
   for (int i=0; i<a.length; i++)
      System.out.println (a[i]); 
}

 

3、

package com.test;

import java.lang.reflect.Array;
import java.util.Arrays;

public class ArrayUtils {

	public static <T> T[] newArrayByArrayClass(Class<T[]> clazz, int length) {  
		return (T[]) Array.newInstance(clazz.getComponentType(), length);  
	}  

	public static <T> T[] newArrayByClass(Class<T> clazz, int length) {  
		return (T[]) Array.newInstance(clazz, length);  
	}  

	public static void main(String[] args) {  
		My[] byTwo = null;
		byTwo = newArrayByClass(My.class, 10);//第一种方法
		for(int i = 0;i < 10; i ++){
			//byTwo[i]=new My();
			try {
				byTwo[i]=My.class.newInstance();
			} catch (InstantiationException e) {
				e.printStackTrace();
			} catch (IllegalAccessException e) {
				e.printStackTrace();
			}
			byTwo[i].aaa =i;
		}
		System.out.println("byTwo:" + byTwo);
		
		
		My[] byOne =null; 
		byOne = newArrayByArrayClass(My[].class, 10);//第二种方法
		for(int i = 0;i < 10; i ++){
			//byOne[i]=new My();
			try {
				byOne[i]=My.class.newInstance();
			} catch (InstantiationException e) {
				e.printStackTrace();
			} catch (IllegalAccessException e) {
				e.printStackTrace();
			}
			byOne[i].aaa =i;
		}
		System.out.println("byOne:" + byOne);
		
		
		String[] sArr = newArrayByClass(String.class, 10);
		for(int i = 0;i < 10; i ++){
			sArr[i] =String.valueOf(i);
		}
		Array.set(sArr, 1, "test");
		System.out.println(Arrays.toString(sArr));
		//[0, test, 2, 3, 4, 5, 6, 7, 8, 9]
		
	}  	
}

class My 
{
	public int aaa;
	public My(){}
}

 。。

 

 

 

 

分享到:
评论

相关推荐

    scrcpy投屏 AssertionError: java.lang.reflect.InvocationTargetExcep

    标题中的问题“scrcpy投屏 AssertionError: java.lang.reflect.InvocationTargetException”是用户在尝试使用Scrcpy时遇到的一个常见错误。这个错误通常意味着在执行某个方法时,Java运行时环境遇到了未预期的情况。...

    解决axis2-CodegenWizardPluginBUG- java.lang.reflect.InvocationTargetException

    标题中的“解决axis2-CodegenWizardPluginBUG- java.lang.reflect.InvocationTargetException”指的是在使用Apache Axis2的CodegenWizardPlugin工具时遇到的一个错误。这个工具是Axis2框架的一部分,用于自动生成...

    java.lang.reflect.Proxy 学习资料 讲解 例子 源码

    java.lang.reflect.Proxy 学习资料 讲解 例子 源码 java.lang.reflect.Proxy 学习资料 讲解 例子 源码 java.lang.reflect.Proxy 学习资料 讲解 例子 源码

    Java.lang.reflect 包下常用的类及方法简介

    在Java编程语言中,`java.lang.reflect`包是核心库的一部分,它提供了运行时访问类、接口、字段和方法的能力。这个包对于理解和操作对象的动态特性至关重要,尤其是在实现反射机制时。反射允许我们在程序运行期间...

    WebLogic错误 java.lang.AssertionError: Registered more than one

    javax.management.remote.JMXServerErrorException: java.lang.AssertionError: Registered more than one instance with the same objectName : com.bea:Name=wl04,Type=UnixMachine new: com.bea:Name=wl04,...

    MyEclipse axis2 wsdl java.lang.reflect.invocationtargetexception

    MyEclipse axis2 wsdl java.lang.reflect.invocationtargetexception code gen 大家要注意一定要仔细,这个问题基本上缺少包引起的,而且一定要clean 如果需要axis2插件 以及这个plugins中的包在我的其他资源里面有

    Java reflect 机制详解

    免费共享,很详细的介绍了反射机制的原理,适合追根究底的java学习者

    commons-lang3-3.1jar:org.apache.commons.lang3.StringUtils等.

    `org.apache.commons.lang3.reflect.FieldUtils`类则是关于反射操作的工具类,它提供了一些安全且方便的方法来访问和修改对象的字段。这些方法包括但不限于: 1. `getField()`:安全地获取类或对象的字段,即使该...

    Android Caused by: java.lang.ClassNotFoundException解决办法

    "Android Caused by: java.lang.ClassNotFoundException解决办法" 在 Android 开发中,ClassNotFoundException 是一个常见的异常,它通常发生在应用程序启动或运行时。这个异常的出现告诉我们,Java 虚拟机无法找到...

    BEWINDOWEB#bewindoweb.github.io#00078-编写basedao的时候报错:java.lang.C

    编写basedao的时候报错:java.lang.Class cannot be cast to java.lang.reflect.Parameterized

    jaxen.jar和dom4j.jar

    at java.lang.reflect.Method.invoke(Unknown Source) at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44) at org.junit.internal.runners.model.ReflectiveCallable.run...

    axis2 InvocationTargetException

    标题 "axis2 InvocationTargetException" 描述的是一个与Apache Axis2框架相关的编程问题,该问题通常在执行服务调用时出现,提示"InvocationTargetException"。这可能是由于多种原因引起的,包括但不限于错误的服务...

    API规范JAVA.pdf

    * java.lang.reflect 是 Java 2 Platform 的一部分,提供了类和接口,以获得关于类和对象的反射信息。 20、数学运算:java.math 提供用于执行任意精度整数算法 (BigInteger) 和任意精度小数算法 (BigDecimal) 的类...

    aop面向切面需要的jar包

    Caused by: java.lang.ClassNotFoundException: org.aspectj.weaver.reflect.ReflectionWorld$ReflectionWorldException at java.net.URLClassLoader$1.run(URLClassLoader.java:200) at java.security.Access...

    JAVA反射实现和实现机制

    5. **`java.lang.reflect.Modifier`**:提供对成员的修饰符的访问。 ##### 基本使用方法 1. **获取成员方法信息**: - `Method[] getMethods()`:获取公开的方法。 - `Method getMethod(String name, Class&lt;?&gt;.....

    wsdl生成java客户端报错

    java.lang.reflect.invocationtargetexception问题的解决方法之一

    java 请求的方法不存在(NoSuchMethodException)

    3. **Java动态代理**:基于`java.lang.reflect.Proxy`和`java.lang.reflect.InvocationHandler`,实现对目标对象的代理,用于拦截方法调用。 综上所述,`NoSuchMethodException`是Java编程中需要关注的一个重要异常...

    java反射(reflect)

    1. 动态代理:使用`java.lang.reflect.Proxy`和`java.lang.reflect.InvocationHandler`可以创建动态代理对象,实现AOP(面向切面编程)。 2. 数据库操作框架:如MyBatis,通过反射动态生成SQL语句并执行。 3. XML...

Global site tag (gtag.js) - Google Analytics