1

从 Java6 切换到 Java8 后,我遇到了一个问题。

突然一个

java.lang.ClassCastException: java.base/java.lang.Integer 不能转换为 java.base/[Ljava.lang.Object;

被抛出。

似乎泛型类型无法与 java.lang.reflect.Method.invoke(Object obj, Object... args) 的 varargs 参数匹配。

由于代码编译干净并且只会在运行时失败,这对我来说看起来真的很可怕。顺便说一句,如果源/目标定义 < 1.8 ,即使使用 java8 编译和执行也会成功!

为了更好地理解,我准备了一个小 JUnit 示例:

public class GenericVarArgTest {
private static InnerClass myInnerClass;

@Test
public void testMethodInvocation()
        throws IllegalAccessException, IllegalArgumentException, InvocationTargetException {

    doSomething();

    System.out.println("Done .. all fine");
}

private void doSomething() throws IllegalAccessException, IllegalArgumentException, InvocationTargetException {
    myInnerClass = this.new InnerClass();
    Method[] methods = myInnerClass.getClass().getMethods();
    for (Method aktMethod : methods) {
        if (aktMethod.getName().equals("printInteger")) {

            // this will always be okay
            // Integer myParam = getSomething();
            // aktMethod.invoke(myInnerClass, myParam);

            // this is just ok, for source / target level <= 1.7 
            aktMethod.invoke(myInnerClass, getSomething());
        }
    }
}

public static <T> T getSomething() {

    final T result = (T) Integer.valueOf(1);

    System.out.println("getSomething will return <T> : " + result);
    return result;
}

public class InnerClass {

    public String printInteger(final Integer aInteger) {
        final String result = "Integer: " + aInteger;
        System.out.println("Will return : " + result);
        return result;
    }
}

我已经使用这个 pom 用 maven 设置了这个小测试:

<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>de.test.java</groupId>
<artifactId>compiler-test</artifactId>
<version>0.0.1-SNAPSHOT</version>
<name>compiler test</name>
<description>Test the different compiler results    </description>

<properties>
    <java.version>1.8</java.version>

    <project.java.source.level>${java.version}</project.java.source.level>
    <project.java.target.level>${java.version}</project.java.target.level>
</properties>


<build>
    <plugins>
        <plugin>
            <artifactId>maven-compiler-plugin</artifactId>
            <version>3.3</version>
            <configuration>
                <source>${project.java.source.level}</source>
                <target>${project.java.target.level}</target>
            </configuration>
        </plugin>
    </plugins>
</build>


<dependencies>
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.12</version>
        <scope>test</scope>
    </dependency>
</dependencies>

如果你执行这个 pom

mvn clean 编译测试 -Djava.version= 1.6

一切都很好。但是执行它

mvn clean 编译测试 -Djava.version= 1.8

将导致 ClassCastException。

任何提示为什么这种行为已经改变。这是一个错误(还是 1.8 之前的一个错误)?非常感谢任何解释。

4

1 回答 1

0

我还没有得出真正的根本原因。但是为了给您提供一些帮助,并非所有 jdk8 版本都存在这种行为。我用jdk 8u60测试了代码,它运行良好(没有ClassCastException抛出)。但是,在jdk 8u144jdk 8u181(在撰写此答案时可用的最新版本)上对其进行测试会导致ClassCastException. 因此,与此同时,要绕过此问题,您可以考虑降级到较早的 jdk8 版本,该版本不会描述错误行为。

于 2018-07-23T13:57:18.530 回答