可变参数方法的java8generic <t>返回类型坏了吗?

时间:2018-07-23 13:16:32

标签: generics javacompiler

从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干净编译测试-Djava.version = 1.6

一切都很好。 但是用

执行它
  

MVN干净编译测试-Djava.version = 1.8

将导致ClassCastException。

任何提示为何此行为已更改。这是错误(还是直到1.8为止的错误)? 非常感谢您的解释。

1 个答案:

答案 0 :(得分:0)

对于实际的根本原因,我仍未得出结论。但是只是为了我可以为您提供一些帮助,并非在所有jdk8版本中都存在此行为。我用 jdk 8u60 测试了代码,它工作正常(未抛出ClassCastException)。但是,在 jdk 8u144 jdk 8u181 (在撰写此答案时可用的最新版本)上进行了测试,结果为ClassCastException。因此,与此同时,要绕过此问题,您可以考虑将其降级到较早的jdk8版本,而该版本没有描述错误的行为。