在Java8之前的版本中,接口中只能声明常量和抽象方法,接口的实现类中必须实现接口中所有的抽象方法。而在Java8中,接口中可以声明默认方法和静态方法,本文,我们就一起探讨下接口中的默认方法和静态方法。
Java 8中允许接口中包含具有具体实现的方法,该方法称为“默认方法”,默认方法使用 default 关键字修饰 。
例如,我们可以定义一个接口MyFunction,其中,包含有一个默认方法getName,如下所示。
public interface MyFunction<T>{
T get(Long id);
default String getName(){
return "binghe";
}
}
在Java8中,默认方法具有“类优先”的原则。
若一个接口中定义了一个默认方法,而另外一个父类或接口中又定义了一个同名的方法时,遵循如下的原则。
1.选择父类中的方法。如果一个父类提供了具体的实现,那么接口中具有相同名称和参数的默认方法会被忽略。
例如,现在有一个接口为MyFunction,和一个类MyClass,如下所示。
public interface MyFunction{
default String getName(){
return "MyFunction";
}
}
public class MyClass{
public String getName(){
return "MyClass";
}
}
此时,创建SubClass类继承MyClass类,并实现MyFunction接口,如下所示。
public class SubClass extends MyClass implements MyFunction{
}
接下来,我们创建一个SubClassTest类,对SubClass类进行测试,如下所示。
public class SubClassTest{
@Test
public void testDefaultFunction(){
SubClass subClass = new SubClass();
System.out.println(subClass.getName());
}
}
运行上述程序,会输出字符串:MyClass。
2.接口冲突。如果一个父接口提供一个默认方法,而另一个接口也提供了一个具有相同名称和参数列表的方法(不管方法是否是默认方法), 那么必须覆盖该方法来解决冲突。
例如,现在有两个接口,分别为MyFunction和MyInterface,各自都有一个默认方法getName(),如下所示。
public interface MyFunction{
default String getName(){
return "function";
}
}
public interface MyInterface{
default String getName(){
return "interface";
}
}
实现类MyClass同时实现了MyFunction接口和MyInterface接口,由于MyFunction接口和MyInterface接口中都存在getName()默认方法,所以,MyClass必须覆盖getName()方法来解决冲突,如下所示。
public class MyClass{
@Override
public String getName(){
return MyInterface.super.getName();
}
}
此时,MyClass类中的getName方法返回的是:interface。
如果MyClass中的getName()方法覆盖的是MyFunction接口的getName()方法,如下所示。
public class MyClass{
@Override
public String getName(){
return MyFunction.super.getName();
}
}
此时,MyClass类中的getName方法返回的是:function。
在Java8中,接口中允许添加静态方法,使用方式接口名.方法名。例如MyFunction接口中定义了静态方法send()。
public interface MyFunction{
default String getName(){
return "binghe";
}
static void send(){
System.out.println("Send Message...");
}
}
我们可以直接使用如下方式调用MyFunction接口的send静态方法。
MyFunction.send();
如果觉得文章对你有点帮助,请微信搜索并关注「 冰河技术 」微信公众号,跟冰河学习Java8新特性。
最后,附上Java8新特性核心知识图,祝大家在学习Java8新特性时少走弯路。