获取数组组件的类

2022-09-01 08:45:36

如果我有

public <T> doSomething(T[] array)
{
}

我怎样才能得到 ?T.classarray

如果我这样做,那就让我反而这样做。array.getClass()T[].class


答案 1

组件类型

使用这个:

array.getClass().getComponentType()

返回表示数组的组件类型。如果此类不表示数组类,则此方法返回 。Classnull

参考:


安全/不安全的铸造

有没有办法从 getComponentType() 返回的类中强制转换为类,而不会收到编译器警告?

采用此方法:

public <T> void doSomething(final T[] array) throws Exception{
    final Class<? extends Object[]> arrayClass = array.getClass();
    final Class<?> componentType = arrayClass.getComponentType();
    final T newInstance = (T) componentType.newInstance();
}

以下是生成的字节码:

public void doSomething(java.lang.Object[] array) throws java.lang.Exception;
     0  aload_1 [array]
     1  invokevirtual java.lang.Object.getClass() : java.lang.Class [21]
     4  astore_2 [arrayClass]
     5  aload_2 [arrayClass]
     6  invokevirtual java.lang.Class.getComponentType() : java.lang.Class [25]
     9  astore_3 [componentType]
    10  aload_3 [componentType]
    11  invokevirtual java.lang.Class.newInstance() : java.lang.Object [30]
    14  astore 4 [newInstance]
    16  return

如您所见,参数类型被擦除到 Object[],因此编译器无法知道 T 是什么。是的,编译器可以使用 ,但这有时会失败得很惨,因为你可以做这样的事情:array.getClass().getComponentType()

Object[] arr = new String[] { "a", "b", "c" };
Integer[] integerArray = (Integer[]) arr;
doSomething(integerArray);

(在本例中返回 ,但代表 。是的,这是合法的,不会生成编译器警告。array.getClass().getComponentType()String.classTInteger


答案 2

如果要对多维数组执行此操作,则以下递归代码将起作用

public static Class<?> getArrayType(Object array) {
    Object element = Array.get(array, 0);
    if (element.getClass().isArray()) {
        return getArrayType(element);
    } else {
        return array.getClass().getComponentType();
    }
}