Java继承的父类实现了某个接口,同时子类也实现该接口的写法

今天在看jdk源码的时候,看到LinkedHashMap类,继承了HashMap,同时实现了Map接口。代码如下

public class LinkedHashMap<K,V>
    extends HashMap<K,V>
    implements Map<K,V>

由于HashMap也继承了Map接口,这就发生了一个有趣的事情。看上去LinkedHashMap通过继承父类的实现,以及自己的直接实现,重复实现了Map接口。

我比较好奇,这样写有什么特殊的作用吗?

从JVM的角度理解,当一个子类继承了父类,或者实现了父接口时,我们可以把这个类的对象当做父类或者父接口的对象来访问,这样就把其他属性屏蔽了,只能访问到引用类型的属性和方法。实现这种功能的原理是,在JVM堆中创建的对象中,包含了一个对象头信息,在对象头中,有一个类型指针,叫Klass Pointer(对象头指针),指向对象所属类型的Class对象的实例,在类型的Class实例中(后面称之为类元信息),记录了该类型的父类,以及实现的所有接口。而在父类的类元信息中,记录了父类的父类,以及父类的所有实现的接口。以此类推,即可知道某个类继承了哪些类,实现了哪些接口。

当我们以某个父类型或者父接口为引用访问某个对象时,先从对象头中找到它的类元信息,再根据类元信息找它实现的所有接口以及父类,如果没有在到父类中去找,以此类推,递归查询,直到查完所有的父类和接口。

所以不管是子类直接实现接口,还是通过父类间接实现接口,对于子类本身来说,都拥有了该接口提供的能力,对于使用该类的程序员来说,功能都是一样的。如果从字节码的角度来看,这样做的结果是,子类的类元信息和父类的类元信息中,在实现接口的列表中都拥有了该接口,如果不这样写,则子类的类元信息中没有该接口信息。

总结一下,这种写法在功能上没有多大影响,也许能够起到强调子类实现了某一接口的作用。但在类的元数据中有小小区别,子类的元数据显式的记录了该实现的接口。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。