Java8新特性:接口的默认方法与接口的静态方法
默认方法允许接口方法定义默认实现,子类方法不必须实现此方法而就可以拥有该方法及实现。如下:
public interface DefaultFuncInter { int getInt(); default String getString(){ return "Default String"; } }
默认方法的优势
默认方法主要优势是提供了一种扩展接口的方法,而不破坏现有代码。如果一个已经投入使用的接口需要扩展一个新的方法,在JDK8以前,我们必须再该接口的所有实现类中都添加该方法的实现,否则编译会出错。如果实现类数量很少且我们有修改的权限,可能工作量会少,但是如果实现类很多或者我们没有修改代码的权限,这样的话就很难解决了。而默认方法提供了一个实现,当没有显式提供时就默认采用这个实现,这样新添加的接口就不会破坏现有的代码。
默认方法另一个优势是该方法是可选的,子类可以根据不同的需求而且经override或者采用默认实现。例如我们定义一个集合几口,其中有增、删、改等操作,如果我们的实现类90%都是以数组保存数据,那么我们可以定义针对这些方法给出默认实现,而对于其他非数组集合或者有其他类似业务,可以选择性复写接口中默认方法。(由于接口不允许有成员变量,所以本示例旨在说明默认方法的优势,并不具有生产可能性)具体参照如下代码:
/** * 定义接口,并包含默认实现方法 */ public interface CollectionDemoInter { //增加默认实现 default void addOneObj(Object object){ System.out.println("default add"); } //删除默认实现 default void delOneObj(Object object){ System.out.println("default del"); } //更新默认实现 default void updateOneObj(Object object){ System.out.println("default del"); } //接口定义需要实现方法 String showMsg(); }
/** * 基于数组的集合实现类,增删改使用默认方法 */ public class Collection4Array implements CollectionDemoInter { @Override public String showMsg() { return null; } }
/** * 特殊集合,不允许删除元素 */ public class NodelCollection implements CollectionDemoInter { @Override public String showMsg() { return null; } @Override public void delOneObj(Object object){ System.out.println("none del"); } }
通过上述代码,大家可以很清楚的发现,如果在接口中定义默认方法,则子类不需要必须实现该默认实现,如果有特殊需求或者需要,则可以Override该实现。
需要注意
- 如果一个类实现两个或两个以上接口,并且多个接口中包含统一默认方法,此时,编译器将报错。这种情况,我们必须让子类Override该方法,否则无法编译通过。
- 在所有的情况,类实现的优先级高于接口的默认实现,也就是先使用自己类中定义的方法或者是父类中的方法。
- 如果是一个接口继承了另外一个接口,2个接口中也包含相同的默认方法,那么继承接口的版本具有更高的优先级。比如A扩展了B接口,那么优先使用A类里面的test方法。
- 通过使用super,可以显式的引用被继承接口的默认实现,语法如下:InterfaceName.super.methodName()。
接口中的静态方法
java8中为接口新增了一项功能:定义一个或者更多个静态方法。类似于类中的静态方法,接口定义的静态方法可以独立于任何对象调用。所以,在调用静态方法时,不需要实现接口,也不需要接口的实例,也就是说和调用类的静态方法的方式类似。语法如:接口名字.静态方法名。
interface A { static String getName() { return "接口A。。。"; } } public class Test implements A { public static void main(String[] args) { System.out.println(A.getName()); } }
注意,实现接口的类或者子接口不会继承接口中的静态方法。static不能和default同时使用。在java8中很多接口中都增加了静态方法,比如下面代码:
public class Test { public static void test(List<String> list) { //直接使用Comparator的静态方法 list.sort(Comparator.comparing(String::length)); } public static void main(String[] args) { List<String> list = Lists.newArrayList("122","2","32"); test(list); for (String str : list) { System.out.println(str); } } }