我的界面基本上是这样的:
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类或完整的堆栈跟踪,我将它们放在外部,因为它们太空了:
答案 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()