java – 转换为在运行时确定的类

我有一个方法fetchObjects(String),期望返回一个Contract业务对象数组. className参数告诉我应该返回什么样的业务对象(当然这在这个解释的情况下没有意义,因为我已经说过我将返回合同,但它基本上是我在真实场景中的情况).所以我从某处获取了一组条目并加载了集合条目的类(其类型由className指定).

现在我需要构造要返回的数组,所以我使用Set的toArray(T [])方法.使用反射,我构建了一个空的Contracts数组.但是,这给了我静态类型Object的值!接下来我需要将它转换为适当的类型,在本例中是Contract [](参见下面列表中的“带星号 – 下划线”部分).

我的问题是:有没有办法,如何,如同在列表中那样转换为Contract [],但仅通过className(或entriesType)确定数组元素的类型(Contract)?换句话说,我想做的基本上是这样的:(entriesType [])valueWithStaticTypeObject,其中entriesType被classname参数指定的类替换,即Contract.

这在某种程度上是不可能的,或者它可以以某种方式完成吗?也许使用泛型?

package xx.testcode;

import java.util.HashSet;
import java.util.Set;

class TypedArrayReflection {

    public static void main(String[] args) {
        try {
            Contract[] contracts = fetchObjects("Contract");
            System.out.println(contracts.length);
        } catch (ClassNotFoundException e) {}
    }

    static Contract[] fetchObjects(String className) throws ClassNotFoundException {
        Class<?> entriesType = Class.forName("xx.testcode."+className);
        Set<?> entries = ObjectManager.getEntrySet(className); 
        return entries.toArray( 
                (Contract[]) java.lang.reflect.Array.newInstance(
                /********/          entriesType, entries.size()) );
    }
}

class Contract { } // business object

class ObjectManager {
    static Set<?> getEntrySet(String className) {
        if (className.equals("Contract"))
            return new HashSet<Contract>();
        return null; // Error
    }
}

谢谢.

更新:使用类型安全方法toArray,取自CodeIdol,我更新了我的fetchObjects方法:

static Contract[] fetchObjects(String className) throws ClassNotFoundException {
    Class<?> entriesType = Class.forName("xx.testcode."+className);
    Set<?> entries = ObjectManager.getEntrySet(className);
    return toArray(entries, entriesType); // compile error
    // -> "method not applicable for (Set<capture#3-of ?>, Class<capture#4-of ?>)"
}

public static <T> T[] toArray(Collection<T> c, Class<T> k) {
    T[] a = (T[]) java.lang.reflect.Array.newInstance(k, c.size());
    int i = 0;
    for (T x : c)
        a[i++] = x;
    return a;
}

我需要做些什么才能摆脱评论中引用的编译器错误?我是否必须指定Set< Contract>在我的getEntrySet方法的返回类型,以便这可以工作?谢谢你的任何指示.

最佳答案
您可以使用该类作为参数,而不是类名.

   static <T extends Contract> T[] buildArray(Class<T> clazz){
     ArrayList<T> l=new ArrayList<T>();
     return l.toArray((T[]) java.lang.reflect.Array.newInstance(clazz, l.size()));
   }

编辑:(读杨评论后)

不,您不能将泛型类型与变量的值一起使用.

转载注明原文:java – 转换为在运行时确定的类 - 代码日志