该教程是针对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()获取所需组件类型的类,然后调用特定的构造函数来初始化数组的每个组件,并设置相应的数组值。