使用varargs和泛型时出现ClassCastException

mur*_*rgo 5 java generics android variadic-functions classcastexception

我正在使用java泛型和varargs.

如果我使用下面的代码,我会得到一个ClassCastException,即使我根本不使用强制转换.

更奇怪的是,如果我在Android上运行它(dalvik),异常中不包含堆栈跟踪,如果我将接口更改为抽象类,则异常变量e为空.

代码:

public class GenericsTest {
    public class Task<T> {
        public void doStuff(T param, Callback<T> callback) {
            // This gets called, param is String "importantStuff"

            // Working workaround:
            //T[] arr = (T[]) Array.newInstance(param.getClass(), 1);
            //arr[0] = param;
            //callback.stuffDone(arr);

            // WARNING: Type safety: A generic array of T is created for a varargs parameter
            callback.stuffDone(param);
        }
    }

    public interface Callback<T> {
        // WARNING: Type safety: Potential heap pollution via varargs parameter params
        public void stuffDone(T... params);
    }

    public void run() {
        Task<String> task = new Task<String>();
        try {
            task.doStuff("importantStuff", new Callback<String>() {
                public void stuffDone(String... params) {
                    // This never gets called
                    System.out.println(params);
                }});
        } catch (ClassCastException e) {
            // e contains "java.lang.ClassCastException: [Ljava.lang.Object; cannot be cast to [Ljava.lang.String;"
            System.out.println(e.toString());
        }
    }

    public static void main(String[] args) {
        new GenericsTest().run();
    }
}
Run Code Online (Sandbox Code Playgroud)

如果你运行它,你会得到一个ClassCastExceptionObject不能转换为String带有堆栈跟踪指向无效的行号.这是Java中的错误吗?我已经在Java 7和Android API 8中测试了它.我为它做了解决方法(在doStuff-method中注释掉了),但是这样做似乎很愚蠢.如果我删除varargs(T...),一切正常,但我的实际实现有点需要它.

来自例外的Stacktrace是:

java.lang.ClassCastException: [Ljava.lang.Object; cannot be cast to [Ljava.lang.String;
    at GenericsTest$1.stuffDone(GenericsTest.java:1)
    at GenericsTest$Task.doStuff(GenericsTest.java:14)
    at GenericsTest.run(GenericsTest.java:26)
    at GenericsTest.main(GenericsTest.java:39)
Run Code Online (Sandbox Code Playgroud)

gra*_*rks 9

这是预期的行为.在Java中使用泛型时,对象的实际类型不包含在已编译的字节码中(这称为类型擦除).将所有类型Object转换并将强制转换插入到已编译的代码中以模拟键入的行为.

另外,varargs成为数组,当调用泛型varargs方法时,Java Object[]在调用之前使用方法参数创建一个类型的数组.

因此,您的行callback.stuffDone(param);编译为callback.stuffDone(new Object[] { param });.但是,您的回调实现需要一个类型数组String[].Java编译器在您的代码中插入了一个不可见的强制转换来强制执行此类型操作,并且由于Object[]无法强制转换String[],因此会出现异常.您看到的虚假行号可能是因为演员表没有出现在代码中的任何位置.

一种解决方法是从Callback接口和类中完全删除泛型,替换所有类型Object.