我如何使这个Java 7兼容?

时间:2015-04-19 11:34:30

标签: java maven generics java-8 java-7

我的界面基本上是这样的:

public interface ISetting<T> {
    public T getDefault();
    public T value();
    public void set(T value);
    public String getName();

    public default String getValueName() {
        Object obj = value();
        if (obj instanceof Boolean) {
            return (boolean)obj ? "Yes" : "No"; 
        }

        return obj.toString();
    }
}

然后在另一个班级我有一个ISetting<?>

的列表
private List<ISetting<?>> settings = Arrays.asList(
        new ClassMode(),
        new EndMode(),
        new PlayerLives(),
        new JoinMidGame(),
        new ScoreboardDisplay(),
        new LifePerKill(),
        new ExplosiveBullets(),
        new ReloadTime());

这一切都完美无缺!但是,我使用我的代码的平台并不支持Java 8,所以我必须使用Java 7,以及问题所在的地方。

如果我将Maven目标设置为1.7,就像在我的pom.xml中一样:

<configuration>
    <source>1.8</source>
    <target>1.7</target>
</configuration>

然后代码完美编译,没有错误或任何东西。但是,当我尝试运行代码时,它给了我这个错误:

  

java.lang.ClassFormatError:类中的方法getValueName   net / uniqraft /谋杀/匹配/设置/ ISetting有非法修饰符:0x1

我尝试使用谷歌,但无法找到我理解或似乎适用于我的任何内容。

所以,我想,我只是将整个代码库变成Java 7:

<configuration>
    <source>1.7</source>
    <target>1.7</target>
</configuration>

我看到的第一个错误是:

  

仅在源级别1.8或以上

允许使用默认方法

这令人非常恼火,我不知道如何绕过它。我的很多代码都依赖于默认实现。我想我只需要使用抽象类?

但我看到的问题更多的是List<Setting<?>>我有:

  

Type mismatch: cannot convert from List<ISetting<? extends Object&Comparable<?>&Serializable>> to List<ISetting<?>>

我不知道这意味着什么或如何解决它。 Eclipse提供的quickfix没有任何帮助。

如果您需要查看完整的非剥离ISetting类或完整的堆栈跟踪,我将它们放在外部,因为它们太空了:

2 个答案:

答案 0 :(得分:8)

我将把答案分成两部分,第一部分关于类型推断,第二部分关于默认方法:

类型推断

在Java 7中,表达式的类型相同,无论上下文。所以当你这样做时:

Arrays.asList(new ClassMode(), new EndMode(), ...);

它不会创建List<ISetting<?>>。您可以将settings类型更改为List<? extends ISetting<?>>,以使其有效。也就是说,一个列表可以包含可以是ISetting<?>或其任何子类型的元素:

List<? extends ISetting<?>> settings = Arrays.asList(new ClassMode(), new EndMode(), ...);

在Java 8中,由于 poly expressions ,将结果列表分配给List<ISetting<?>>。这意味着某些表达式的推导类型可能会受到目标类型的影响。所以当你这样做时:

private List<ISetting<?>> settings = Arrays.asList(new ClassMode(), new EndMode(), ...);

编译器分析目标类型并隐式将类型参数传递给Arrays.asList(),这相当于:

private List<ISetting<?>> settings = Arrays.<ISetting<?>>asList(new ClassMode(), new EndMode(), ...);

创建List<ISetting<?>>并将其分配给settings。如果您不想更改settings类型,上述表单也适用于Java 7。

默认方法

Java 7没有默认方法。相反,您可以创建一个抽象的骨架实现来与您的界面一起使用。接口将定义类型,骨架实现将提供默认功能。

首先,将界面中的default方法转换为常规方法声明:

public interface ISetting<T> {
    T getDefault();
    T value();
    void set(T value);
    String getName();

    // Former default methods:
    String getValueName();
    boolean isHidden();
    boolean isDefault();
    // etc.
}

然后创建一个抽象类来保存默认实现:

public abstract class AbstractSetting<T> implements ISetting<T> {

    @Override
    public String getValueName() {
        Object obj = value();
        if (obj instanceof Boolean) {
            return ((Boolean) obj) ? "Yes" : "No";
        }
        return obj.toString();
    }

    @Override
    public boolean isHidden() {
        return false;
    }

    // etc.
}

现在让你的具体类实现ISetting<T>接口并扩展AbstractSetting<T>类。例如:

public class ConcreteSetting extends AbstractSetting<Boolean> implements ISetting<Boolean> {
    // concrete implementation
}

答案 1 :(得分:5)

您需要执行以下操作之一:

  • getValueName()
  • 中删除默认修饰符
  • make [abstract] class而不是interface
  • 将您的平台升级到Java 1.8