实现类与接口之间首先是子类型关系,同时从设计意图上也可以理解为 can-do 关系。两者并不矛盾,但"子类型关系"是更底层、更精确的类型系统描述。
1. 从 Java 类型系统角度:子类型关系
- 在 Java 中,如果一个类
C实现了接口I,那么C类型的对象可以赋值给I类型的变量:
I obj = new C(); - 这符合子类型 (subtype)的定义:
C是I的一个子类型。 - 子类型关系保证了里氏替换原则(Liskov Substitution Principle):凡是父类型(接口)出现的地方,都可以用子类型(实现类)替换。
因此,从语言规范和类型理论来看,实现类与接口之间是子类型关系。
2. 从设计意图角度:can-do 关系
- 接口通常用于定义一组行为或能力,如
Runnable、Comparable、Serializable。 - 当说"一个类实现了
Runnable",意味着"这个类可以运行"(can do run)。 - 这种视角强调接口是能力的声明,而不是"是什么"(is-a)。
- 所以很多教材和设计原则中,接口被描述为 can-do 关系,以区别于类继承的 is-a 关系。
3. 两者如何统一?
- 子类型关系是 can-do 关系在类型系统中的具体实现形式。
- 可以这样理解:
- 继承(
extends)表达的是 is-a(是一种)------ 子类是父类的一种特殊类型。 - 实现(
implements)表达的是 can-do(能做什么)------ 但 Java 要求这种能力必须通过子类型关系来体现,因此实现类自然成为接口的子类型。
- 继承(
- 换句话说:can-do 是语义,子类型是语法和类型约束。
4. 为什么不能只说 can-do?
- 仅仅说 can-do 会忽略类型系统的重要特性:多态、替换性、类型检查。
- 例如,一个方法声明
void sort(Comparable c),它接受任何实现了Comparable的子类型。如果只用 can-do 描述,无法解释为什么可以传递String或自定义类------这背后是子类型关系在起作用。
总结
实现类与接口之间,严格来说是子类型关系(Java 类型系统层面),同时从设计意图上可以理解为 can-do 关系。两者结合才能完整描述:接口既定义了一组能力(can-do),又在类型系统中为其所有实现类建立了一个公共父类型(子类型关系)。
在实际教学和沟通中,如果需要强调类型替换,说"子类型";如果需要强调接口的设计作用,说"can-do"。但最准确的回答是:首先是子类型关系,它实现了 can-do 的语义。