java 动态创建对象_通过类名称在Java中动态创建对象并通过使用带有数据的列表来设置类字段...

小编典典

对象的动态实例化可能会变得非常复杂,您的方案涉及以下几个方面:

将对象值从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 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。

找到构造函数后,循环遍历Stringargs将它们转换为构造函数期望的类型。假设可行,然后我们通过反射调用构造函数,挥动魔杖并说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

辉煌

2020-09-18

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值