【Java基础】接口和抽象类之间的对比
java 中的接口和抽象类之间的对比
一、接口
interface
,将其翻译成插座可能就更好理解了。我们通常利用接口来定义实现类的行为,当你将插座上连接笔记本的三角插头拔掉,换成微波炉插上去的时候,你就会发现,这两样东西它都是三角插头的。那么这个三角插头就可以视为一种规则,而这两样电器就是两个实现了同样规则的构件了。因为实现了同样的规则,使得动态地将一个构件换成另外一个构件变得容易得多。那么在代码中也是相同地道理,当两个类实现了相同的接口,将客户端中原有实现类换成另外一个,就变得简单不过了。
首先,定义一个接口的代码如下:
[public] interface interfacename { void fun1(); }
接口中可以拥有变量和方法,但是接口中定义的变量会被隐式默认为 public static final
变量,并且也只能是这样,如果你使用其他的修饰符修饰,编译时会报错。同样的,方法也会默认被 public abstract
修饰。但是,你只能定义一个方法,而不能有方法的实现类,接口中的方法只能是抽象方法,这一点就和抽象类不同了(抽象类可以有方法实现)。
而接口对应实现类的实例代码如下
public class classname implements interfacename, otherinterface, [...] { public void fun1() { // do something ... } }
在 java 中,类是单继承的,但是却可以继承多个接口,而接口可以继承多个接口。
二、抽象类
在 java 语言中,类有两种:一种是具体类,另一种是抽象类。具体类可以实例化,抽象类不可以实例化。所以在这里可以想到,抽象类创建出来,就是用来被继承的,毕竟你不能用一个无法实例化的类来为你完成什么功能(当然这里不包括静态变量和方法的调用,但是如果只是用来做这些,那你为什么要把这个类声明为抽象类呢?)
在了解抽象类之前,我们先了解抽象方法:抽象方法是一种特殊的方法,它只有声明,而没有具体的实现:
abstract void fun1();
在抽类中的定义的抽象方法必须使用 absract
关键字修饰,同样抽象类也需要被 abstract
关键字修饰
[public] abstract class abstractclassname { abstract void fun1(); }
如果一个类继承了抽象类,那么子类就必须要实现抽象类中定义的所有抽象方法,除非将子类也定义为抽象类。
三、对比
语法功能上的对比
- 接口只能包含抽象方法,而抽象类即可有普通方法,也能有抽象方法
- 接口不包含构造方法,抽象类中可以包含构造方法以备继承类扩充
- 接口中不能含有静态代码块以及静态方法,而抽象类可以有静态代码块和静态方法
- 接口只能定义静态常量属性,抽象类既可以定义普通属性,也可以定义静态常量属性
设计思想上的对比
我们口头上常说:实现一个接口,继承一个抽象类。其实这句话就已经将接口和抽象类之间的区别表现出来了。
接口更多的是被视为一种契约,契约里的方法是让你来实现的。当别人给定你一个接口,你就按接口中定义的方法去实现,那么就是在实现一份契约了。所以接口是其实是一些方法特征的集合,这些方法特征当然来自于具体方法,但是它们一般都是来自于一些在系统中不断出现的方法。一个接口只有方法的特征,而没有方法的实现,因此这些方法在不同的地方被实现的时候,可以具有不同的行为。
而抽象类更多的是被视为一个模板,它提供的是一个继承的出发点,是子类中的共有部分的集中实现。所以抽象类是一种模板式的设计,什么是模板式设计?最简单例子,大家都用过 ppt 里面的模板,如果用模板a设计了 ppt b和 ppt c,ppt b 和 ppt c 公共的部分就是模板a了,如果它们的公共部分需要改动,则只需要改动模板a就可以了,不需要重新对 ppt b 和 ppt c 进行改动。也就是说对于抽象类,如果需要添加新的方法,可以直接在抽象类中添加具体的实现,子类可以不进行变更;而对于接口则不行,如果接口进行了变更,则所有实现这个接口的类都必须进行相应的改动。
四、继承复用与规范实现之间的选择
接口是一种特殊的抽象类,那么在开发中,何时选用接口?何时选用抽象类呢?
- 优先选用接口
满足以下全部条件时,选用抽象类
子类是父类的一个特殊类,而不是父类的一个角色,也就是要区分 “has-a” 与 “is-a” 两种关系的不同。has-a 关系应当使用聚合关系描述,而只有 is-a 关系才符合继承关系。
不会出现需要将子类换成另外一个类的子类的情况
子类具有扩展父类的责任。而不是具有置换掉(override)或注销掉(nullify)父类的责任。如果子类需要大量地置换掉父类的行为,那么这个子类就不应当成为这个父类的子类。
只有父类和子类属于同一种分类的时候,才可以使用继承,不要从工具类继承。
五、java 8 的新特性
从 java 8 开始,接口也可以拥有默认的方法实现,这是因为不支持默认方法的接口的维护成本太高了。在 java 8 之前,如果一个接口想要添加新的方法,那么要修改所有实现了该接口的类。
public interface interfaceexample { void func1(); default void func2(){ system.out.println("func2"); } }
public class interfaceimplementexample implements interfaceexample { @override public void func1() { system.out.println("func1"); } }
public static void main(string[] args) { interfaceexample example = new interfaceimplementexample(); example.fun1(); example.fun2(); }
六、参考
- 《java与模式》
- https://www.cnblogs.com/felixzh/p/5938544.html
- https://www.cnblogs.com/devinzhang/archive/2011/12/24/2300260.html
- https://github.com/cyc2018/cs-notes/blob/master/notes/java%20%e5%9f%ba%e7%a1%80.md#%e6%8a%bd%e8%b1%a1%e7%b1%bb%e4%b8%8e%e6%8e%a5%e5%8f%a3