抽象类和接口的区别
在面向对象编程当中,抽象类和接口是为抽象而生而的两个概念,在初学时特别容易搞混它们俩。
Java 既支持接口,也支持抽象类,这里主要拿 Java 的接口和抽象类做比较。简单地在 Java 中定义这两个概念就是,抽象类是包含抽象方法的类,接口是对行为的抽象。
抽象类
在 Java 中,抽象类仍然以 class
定义,并在此基础上增加 abstract
修饰,如下是抽象类的定义:
[public|protected] abstract class ClassName { abstract void fun(); }
从定义上看,Java 中的抽象类就是用来继承的,没有被继承的抽象类没有任何实际的作用。而且,抽象类中的抽象方法只是起到一个限制的作用,并没有提供实际的方法体,这也要求子类去实现自己的方法体。
将抽象类的特征总结一下,大概有以下几点:
- 抽象类不允许被实例化,只能被继承
- 抽象类可以包含属性和方法,方法里既可以包含具体实现,也可以不包含具体实现,不包含具体实现的方法称为抽象方法
- 子类继承抽象类,必须实现抽象类的所有抽象方法
接口
在 Java 中,接口以 interface
定义,与 class
定义的类不同,如下是接口的定义:
[public|protected] interface InterfaceName { void func(); }
接口实际上也可以包含变量和方法,但是,接口中的变量会被隐式地指定为 public static final
修饰的不可变量,接口中的方法会被隐式地指定为 public abstract
修饰的方法。
将接口的特性总结一下,大概有以下几点:
- 接口不能声明属性,可以声明的是静态变量
- 接口声明的方法不包含具体实现
- 类实现接口的时候,必须实现接口中声明的所有方法
区别
从上述对抽象类和接口的简单分析看,抽象类和接口的概念非常相似,从明面上看,其最大的区别就是,抽象类是用来继承的,接口是用来实现的。
从更深层次的角度上看,抽象类是不能被实例化的类,只能被子类继承,继承关系表示的是一种 is-A 的关系,接口表示的是一种 has-A 关系。
在使用时,抽象类可以定义一些公共的属性、方法,抽象方法用于声明子类继承的约束;接口的主要作用就是声明实现的协议,但是相比抽象类的优势就是一个类可以实现多个接口。
抽象类和接口的使用
抽象类的使用
首先,只能被子类继承的抽象类能解决代码复用的问题。
然后,抽象类表达的是一种抽象概念,适用于表示现实生活中的抽象概念。如狗是具体对象,动物则是抽象概念。
使用抽象方法,而非空的方法体,创建子类时就知道他必须要重写该方法,而不能忽略。
使用抽象类,类的使用者创建对象的时候,就知道他必须要使用某个具体子类,而不是抽象类本身。
使用抽象类提高了安全性,降低了开发者犯错的概率,是一种更优雅的编码方式。
抽象类更多的作用是引导使用者正确使用,避免被误用。
接口的使用
接口是对行为的一种抽象,相当于一组协议,更侧重于解耦。
调用者只需要关注抽象的接口,不需要了解具体的实现,具体的实现代码对调用者透明。接口实现了约定和实现相分离,可以降低代码间的耦合,提高代码的扩展性。
配合使用
如果抽象类只定义抽象方法,那抽象类和接口非常相似。但接口和抽象类在根本上是不同的,一个类可以实现多个接口,但只能继承一个类。
抽象类和接口是配合而不是替代,它们经常一起使用,接口声明能力,抽象类提供默认实现,实现全部或部分方法,一个接口经常有一个对应的抽象类。
比如,在 Java 类库中有以下关系:
- Collection 接口和对应的 AbstractCollection 抽象类
- List 接口和对应的 AbstractList 抽象类
- Map 接口和对应的 AbstractMap 抽象类
模拟抽象类和接口
通过抽象类实现接口
接口没有成员变量,没有方法实现,只有方法声明,实现接口的类必须实现接口中的所有方法。
只要满足上述几个特点,从设计的角度上讲,它就可以叫作接口。
在 Java 中,使用抽象类实现起来也比较简单,即抽象类只定义抽象方法即可,缺陷就是子类无法继承多个抽象类。
用普通类模拟接口
普通的类是可以包含具体实现的,这不符合接口的定义。但是,可以让类中的方法抛出 NoSuchMethodError
错误来模拟不包含实现的接口,并且强迫子类在继承这个父类时都去主动实现父类的方法,否则就会在运行时抛出异常。
为了避免普通的类被实例化,需要将这个类的构造函数声明成 protected
访问权限。
具体的代码实现如下:
public class MockInterface { protected MockInterface() {} public void funcA() { throw new NoSuchMethodError(); } }
同样的,无论是使用抽象类还是普通类,实现的接口都无法满足接口的所有特性,这里也仅做一些了解。
基于接口而非实现编程
在软件开发中,最大的挑战之一就是需求的不断变化,因此,开发时一定要具有抽象意识、封装意识、接口意识。
越抽象、越顶层、越脱离具体某一实现的设计,越能提高代码的灵活性、扩展性、可维护性。
这个时候,接口的存在就非常必要了,通过使用接口定义实现类的协议,将约定和实现分离,做到了解耦的效果。
在定义接口的时候,一些注意事项就是:命名一定要足够通用,不能包含跟具体实现相关的字眼;另一方面,与特定实现相关的方法不要定义在接口中。
通常,越是不稳定的系统,越是要在代码的扩展性、维护性上下功夫。相反,某个系统特别稳定,在开发完成之后,基本不需要做维护,则没有必要为其扩展性、维护性投入不必要的开发时间。