该教程是针对JDK 8编写的。本页面描述的示例和实践不利用后续版本中引入的改进,并且可能使用不再可用的技术。
有关Java SE 9和后续版本中更新的语言特性的摘要,请参见Java语言变更。
有关所有JDK版本的新功能、增强功能以及已删除或已弃用选项的信息,请参见JDK发行说明。
与非反射代码一样,反射支持通过 java.lang.reflect.Array.newInstance()
动态创建任意类型和维度的数组的能力。考虑一个名为
的基本解释器,它能够动态创建数组。将解析的语法如下所示:ArrayCreator
fully_qualified_class_name variable_name[] = { val1, val2, val3, ... }
假设 fully_qualified_class_name
表示一个具有单个 String
参数的构造函数的类。数组的维度由提供的值的数量确定。下面的示例将构造一个 fully_qualified_class_name
类型的数组实例,并使用 val1
、val2
等实例填充其值。(此示例假设熟悉 Class.getConstructor()
和 java.lang.reflect.Constructor.newInstance()
。有关反射 API 的 Constructor
的讨论,请参阅本教程的 创建新类实例 部分。)
import java.lang.reflect.Array; import java.lang.reflect.Constructor; import java.lang.reflect.InvocationTargetException; import java.util.regex.Pattern; import java.util.regex.Matcher; import java.util.Arrays; import static java.lang.System.out; public class ArrayCreator { private static String s = "java.math.BigInteger bi[] = { 123, 234, 345 }"; private static Pattern p = Pattern.compile("^\\s*(\\S+)\\s*\\w+\\[\\].*\\{\\s*([^}]+)\\s*\\}"); public static void main(String... args) { Matcher m = p.matcher(s); if (m.find()) { String cName = m.group(1); String[] cVals = m.group(2).split("[\\s,]+"); int n = cVals.length; try { Class<?> c = Class.forName(cName); Object o = Array.newInstance(c, n); for (int i = 0; i < n; i++) { String v = cVals[i]; Constructor ctor = c.getConstructor(String.class); Object val = ctor.newInstance(v); Array.set(o, i, val); } Object[] oo = (Object[])o; out.format("%s[] = %s%n", cName, Arrays.toString(oo)); // production code should handle these exceptions more gracefully } catch (ClassNotFoundException x) { x.printStackTrace(); } catch (NoSuchMethodException x) { x.printStackTrace(); } catch (IllegalAccessException x) { x.printStackTrace(); } catch (InstantiationException x) { x.printStackTrace(); } catch (InvocationTargetException x) { x.printStackTrace(); } } } }
$ java ArrayCreator java.math.BigInteger [] = [123, 234, 345]
上面的示例展示了一种可能需要使用反射来创建数组的情况;即在运行时才知道组件类型的情况下。在这种情况下,代码使用Class.forName()
获取所需组件类型的类,然后调用特定的构造函数来初始化数组的每个组件,并设置相应的数组值。