通过类名称在Java中动态创建对象并通过使用带有数据的列表来设置类字段
我有一个包含字符串类型为->的数据的列表 ["classField1", "classField2", "classField3"]
我有一个方法(myMethod(List list, String
className)
)接受列表作为参数。因此,我可以通过参数将此List传递给myMethod(List list,String className)。
在中myMethod
,我想创建一个对象,它将是className的实例,即第二个参数。之后,我想通过使用List的数据来设置类的字段。由于我想动态获取类的字段,因此上述结果是我必须将列表的每个String值强制转换为类的每个字段的类型。
我确定列表内的字符串的顺序是正确的,并且与具有相同顺序的类的字段相对应。
有人知道如何执行上述操作吗?
例:
["StringtempValue", "StringUnitOfMeasurement"]
=>
创建实例对象:
public class TempStruct {
private double tempValue;
private String unitOfMeasurement;
public TempStruct(double tempValue, String unitOfMeasurement) {
this.tempValue = tempValue;
this.unitOfMeasurement = unitOfMeasurement;
}
}
我尝试通过以下方式给出解决方案:
实际上,我想创建一个现有类的对象,并尝试通过反射来实现。我使用以下代码:
Class<?> cls = Class.forName(name);
Object clsInstance = (Object) cls.newInstance();
Field[] objectFields = clsInstance.getClass().getDeclaredFields();
但是当第二行尝试创建新对象时,我遇到了例外。正如@JB Nijet所说,我不知道getDeclaredFields()方法不会返回排序的字段。
实际上,我有一个仅接受字符串列表的方法,因此通过使用反射,我将对象转换为字符串列表,此后,我想做相反的事情。我认为没有其他方法可以做到。
-
对象的动态实例化可能会变得非常复杂,您的方案涉及以下几个方面:
- 将对象值从
String
转换为适当的类型 - 从类名加载正确的类并创建一个实例
- 将这些值分配给对象
毫无疑问地将Java视作一种动态语言,对这些要点的全面讨论将占据整整一章。但是,假设您没有时间学习这些错综复杂的内容,或者依赖一些庞大的第三方库,那么我们就来整理一些有助于您解决问题的方法。请随时将手放在车内,以免颠簸。
让我们首先解决类型转换的问题。该值被设置为
Strings
,但你的对象将它们存储为double
,long
,int
等,所以我们需要解析的功能String
到相应的目标类型:static Object convert(Class<?> target, String s) { if (target == Object.class || target == String.class || s == null) { return s; } if (target == Character.class || target == char.class) { return s.charAt(0); } if (target == Byte.class || target == byte.class) { return Byte.parseByte(s); } if (target == Short.class || target == short.class) { return Short.parseShort(s); } if (target == Integer.class || target == int.class) { return Integer.parseInt(s); } if (target == Long.class || target == long.class) { return Long.parseLong(s); } if (target == Float.class || target == float.class) { return Float.parseFloat(s); } if (target == Double.class || target == double.class) { return Double.parseDouble(s); } if (target == Boolean.class || target == boolean.class) { return Boolean.parseBoolean(s); } throw new IllegalArgumentException("Don't know how to convert to " + target); }
啊。这很丑陋,并且仅处理内部类型。但是我们不是在这里寻求完美,对吧?因此,请适当增强。请注意,从转换
String
为其他类型实际上是反序列化的一种形式,因此您要对客户(无论给您什么Strings
)施加约束,以特定格式提供其值。在这种情况下,格式由parse
方法的行为定义。练习1:在将来的某个时候,以向后不兼容的方式更改格式,以引起某人的愤怒。现在让我们进行实际的实例化:
static Object instantiate(List<String> args, String className) throws Exception { // Load the class. Class<?> clazz = Class.forName(className); // Search for an "appropriate" constructor. for (Constructor<?> ctor : clazz.getConstructors()) { Class<?>[] paramTypes = ctor.getParameterTypes(); // If the arity matches, let's use it. if (args.size() == paramTypes.length) { // Convert the String arguments into the parameters' types. Object[] convertedArgs = new Object[args.size()]; for (int i = 0; i < convertedArgs.length; i++) { convertedArgs[i] = convert(paramTypes[i], args.get(i)); } // Instantiate the object with the converted arguments. return ctor.newInstance(convertedArgs); } } throw new IllegalArgumentException("Don't know how to instantiate " + className); }
我们在这里采取了许多捷径,但是,这不是我们正在创建的西斯廷教堂。只需加载该类并搜索其参数数量与参数数量(即arity)匹配的构造函数。重载了相同的构造函数?不,不行。Varargs?不,不行。非公共构造函数?不,不行。而且,如果您不能保证您的类将提供一个像示例一样设置所有字段的构造函数
TempStruct
,则我将其称为“一天喝啤酒”,因为这种方法是DOA。找到构造函数后,循环遍历
String
args将它们转换为构造函数期望的类型。假设可行,然后我们通过反射调用构造函数,挥动魔杖并说abracadabra。Voilà:您有一个新对象。让我们尝试一个非常人为的例子:
public static void main(String[] args) throws Exception { TempStruct ts = (TempStruct)instantiate( Arrays.asList("373.15", "Kelvin"), TempStruct.class.getName()); System.out.println( ts.getClass().getSimpleName() + " " + ts.tempValue + " " + ts.unitOfMeasurement); }
输出:
TempStruct 373.15 Kelvin
辉煌
- 将对象值从