2

我有以下 Java 类(带有嵌套类/接口)。main从 Eclipse(版本:2019-09 R (4.13.0))中运行该方法时,我得到以下输出:

java.version: 1.8.0_241
PageA.m3() called 

这是 Eclipse 使用的命令行:

/Library/Java/JavaVirtualMachines/jdk1.8.0_241.jdk/Contents/Home/bin/java -agentlib:jdwp=transport=dt_socket,suspend=y,address=localhost:52180 -javaagent:/Users/*****/eclipse/java-2019-09/Eclipse.app/Contents/Eclipse/configuration/org.eclipse.osgi/222/0/.cp/lib/javaagent-shaded.jar -Dfile.encoding=UTF-8 -classpath <a-list-of-.jar-files> _play.PageTest

从 Intellij(IDEA 2019.3.3(社区版))中运行相同的代码时,我得到以下输出:

src/main/java/_play/PageTest.java:13: error: reference to m1 is ambiguous
                .m1(pageAorB -> ((SpecialPage<?>) pageAorB).m3());
                ^
  both method m1(Consumer<T#1>) in Page and method m1(Consumer<T#2>) in AbstractPage match
  where T#1,T#2 are type-variables:
    T#1 extends Page<T#1> declared in interface Page
    T#2 extends Page<T#2> declared in class AbstractPage

为什么我在 Intellij 中出现此错误,但在 Eclipse 中却没有?有没有办法解决这个问题,所以它在 Intellij 中运行没有错误?

这是Java类:

package _play;

import java.util.function.Consumer;
import java.util.function.Function;

public class PageTest {

    public static void main(String[] args) {
        System.out.println("java.version: " + System.getProperty("java.version"));

        new PageC()
                .m2(pageC -> (1 == 1 ? new PageA() : new PageB()))
                .m1(pageAorB -> ((SpecialPage<?>) pageAorB).m3());
    }

    public static interface Page<T extends Page<T>> {
        T m1(Consumer<T> c);
        <R> R m2(Function<? super T, ? extends R> f);
    }

    public static abstract class AbstractPage<T extends Page<T>> implements Page<T>{
        @Override
        public T m1(Consumer<T> c){
            c.accept(self());
            return self();
        }

        @Override
        public final <R> R m2(Function<? super T, ? extends R> f) {
            return f.apply(self());
        }

        abstract protected T self();
    }

    public static interface SpecialPage<T extends Page<T>> extends Page<T> {
        T m3();
    }

    public static class PageC extends AbstractPage<PageC> {

        @Override
        protected PageC self() {
            return this;
        }
    }

    public static class PageB extends AbstractPage<PageB> implements SpecialPage<PageB> {
        @Override
        public PageB m3() {
            System.out.println("PageB.m3() called");
            return this;
        }

        @Override
        public PageB self() {
            return this;
        }
    }

    public static class PageA extends AbstractPage<PageA> implements SpecialPage<PageA> {
        @Override
        public PageA m3() {
            System.out.println("PageA.m3() called");
            return this;
        }

        @Override
        public PageA self() {
            return this;
        }
    }
}

编辑 在这种情况下,页面接口和 AbstractPage 类位于客户端无法更改的库中。但客户端应该能够扩展 Page 接口。

4

1 回答 1

1

问题是,默认情况下 IntelliJ 和 Gradle 使用的Eclipse 编译器(ecj)是 4.13 版还是javac1.8.0_241 版?

的错误消息javac说它是模棱两可的,因为AbstractPage类型变量T指的是AbstractPage(命名T)的类型变量,也指的是Page(也命名T)的不同类型变量。但实际上,两个类型变量都指的是同一个类型,并不是因为它们的名字相同,而是因为AbstractPageimplements Page<T>。它不是模棱两可的,并且javac在这里错误地给出了编译错误。

作为此javac错误的解决方法,您可以执行以下操作之一:

  • 在 IntelliJ 和 Gradle 中也使用 Eclipse 编译器
  • 使用中间结果的变量重写main方法中的代码,以便不需要推断它:SpecialPage<? extends Page<?>>javac

    SpecialPage<? extends Page<?>> pageAorB = new PageC().m2(pageC -> (1 == 1 ? new PageA() : new PageB()));
    pageAorB.m1(specialPage -> ((SpecialPage<?>) specialPage).m3());
    
于 2020-02-13T17:01:30.997 回答