不学Java8你就Out了!——Java8的default方法详解

Java 8新增了default方法,它可以在接口添加新功能特性,而且还不影响接口的实现类。下面我们通过例子来说明这一点。

public class MyClass implements InterfaceA {
	public static void main(String[] args){
	}
	
	@Override
	public void saySomething() {
		// TODO Auto-generated method stub
	}
}

interface InterfaceA{
	public void saySomething();
}

上面的代码显示了MyClass类实现了InterfacesA接口的saySomething()方法。现在我们为InterfacesA接口新增一个sayHi()方法。这么做的话,MyClass类是无法通过编译的,除非我们提供了sayHi()的实现方法。
Default方法是非常有用的,通过在接口定义的方法的访问修饰符前加上关键字default,那么实现类就无需提供该方法的实现了。比如:

public class MyClass implements InterfaceA {
	public static void main(String[] args){
	}
	
	@Override
	public void saySomething() {
		// TODO Auto-generated method stub
	}
}

interface InterfaceA{
	public void saySomething();
	default public void sayHi(){
		System.out.println("Hi");
	}
}

要注意,我们必须提供所有的default方法的实现。因此,default方法使我们的代码更加灵活,在接口中也可以写方法实现了。实现的方法会作为默认的方法实现。

那么,多接口存在冲突该怎么办?

由于Java类可以实现多个接口,那么就可能存在这样的情况:两个或多个接口都有一个同名的default接口方法,从而造成冲突。因为Java虚拟机在程序运行时,并不清楚你要使用哪一个default方法。这会导致编译错误。

让我们来看看下面的例子。

public class MyClass implements InterfaceA, InterfaceB {
	public static void main(String[] args){
		MyClass mc = new MyClass();
		mc.sayHi();
	}
	
	@Override
	public void saySomething() {
		// TODO Auto-generated method stub
	}
}

interface InterfaceA{
	public void saySomething();
	default public void sayHi(){
		System.out.println("Hi from InterfaceA");
	}
}

interface InterfaceB{
	default public void sayHi(){
		System.out.println("Hi from InterfaceB");
	}
}

它是通不过编译的,会报以下错误:

“Duplicate default methods named sayHi with the parameters () and () are inherited from the types InterfaceB and InterfaceA.”

除非在MyClass类中重写了sayHi()方法:

public class MyClass implements InterfaceA, InterfaceB {
	public static void main(String[] args){
		MyClass mc = new MyClass();
		mc.sayHi();
	}
	
	@Override
	public void saySomething() {
		// TODO Auto-generated method stub
	}
	
	@Override
	public void sayHi(){
		System.out.println("implemetation of sayHi() in MyClass");
	}
}

interface InterfaceA{
	public void saySomething();
	default public void sayHi(){
		System.out.println("Hi from InterfaceA");
	}
}

interface InterfaceB{
	default public void sayHi(){
		System.out.println("Hi from InterfaceB");
	}
}

如果想指定调用哪一个接口的sayHi()方法,我们可以这么做:

public class MyClass implements InterfaceA, InterfaceB {
	public static void main(String[] args){
		MyClass mc = new MyClass();
		mc.sayHi();
	}
	
	@Override
	public void saySomething() {
		// TODO Auto-generated method stub
	}
	
	@Override
	public void sayHi(){
		InterfaceA.super.sayHi();
	}
}

interface InterfaceA{
	public void saySomething();
	default public void sayHi(){
		System.out.println("Hi from InterfaceA");
	}
}

interface InterfaceB{
	default public void sayHi(){
		System.out.println("Hi from InterfaceB");
	}
}

答案是不是很简单呢?

?

?

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。