17

我有一个包含字符串类型数据的列表->["classField1", "classField2", "classField3"]

我有一个方法 ( myMethod(List list, String className)) 接受列表作为参数。所以,我可以通过参数将这个 List 传递给 myMethod(List list, String className)。

myMethod中,我想创建一个对象,它将是 className 的实例,即第二个参数。之后,我想通过使用列表的数据来设置类的字段。由于我想动态获取类的字段,上面的结果是我必须将列表的每个 String 值强制转换为类的每个字段的类型。

我确信 List 中的字符串的顺序是正确的,并且对应于具有相同顺序的类的字段。

有人知道如何执行上述操作吗?

例子:

["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() 方法不会返回已排序的字段。

实际上,我有一个只接受字符串列表的方法,所以通过使用反射我将对象转换为字符串列表,然后我想做相反的事情。我没有想到任何其他方法可以做到这一点。

4

3 回答 3

53

对象的动态实例化可能会变得非常复杂,并且您的场景涉及几个方面:

  • 将对象值从转换String为适当的类型
  • 从类名加载正确的类并创建一个实例
  • 将这些值分配给对象

对这些要点的深入讨论将占据一整章,毫无疑问地把 Java 作为一种动态语言进行处理。但是,假设你没有时间学习这些错综复杂的东西,或者依赖于一些庞大的第三方库,让我们来做一些让你上路的事情。请始终将双手放在车内,因为骑行会变得颠簸。

让我们先解决类型转换的问题。这些值提供为Strings,但您的对象会将它们存储为doublelongint等。因此我们需要一个将 a 解析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)匹配的构造函数。重载相同数量的构造函数?不,不会工作。可变参数?不,不会工作。非公共构造函数?不,不会工作。如果你不能保证你的类会像你的例子一样提供一个设置所有字段的构造函数TempStruct,那么我会收工并喝杯啤酒,因为这种方法是 DOA。

一旦我们找到构造函数,就循环遍历Stringargs 以将它们转换为构造函数所期望的类型。假设可行,然后我们通过反射调用构造函数,挥动魔杖并说 abracadabra。瞧:你有一个新对象。

让我们用一个非常人为的例子来试试:

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

辉煌

于 2012-12-14T03:20:48.067 回答
3

我曾经遇到过同样的问题,结果证明 hashMap 是我的解决方案。

看看: http ://docs.oracle.com/javase/6/docs/api/java/util/HashMap.html

于 2012-12-13T21:47:57.563 回答
0

看看http://commons.apache.org/beanutils/包。它允许按名称访问字段。

于 2012-12-13T22:59:44.613 回答