△Hollis, 一个对Coding有着独特追求的人△

这是Hollis的第 227篇原创分享

作者 l Hollis

来源 l Hollis(ID:hollischuang)

从学习Java的第一天起,我们就知道Java是一种面向对象语言,而学习Java的第二天,我们就知道了面向对象的三大基本特性是:封装、继承、多态。

所以,对于很多开发者来说,继承肯定都是不陌生的。但是,继承一定适合所有的场景吗?毫无忌讳的使用继承来做代码扩展真的好吗?

为什么《阿里巴巴Java开发手册》中有一条规定:谨慎使用继承的方式进行扩展,优先使用组合的方式实现。


本文就来针对这些问题,简单分析一下。

1

面向对象的复用技术

每个人在刚刚学习继承的时候都会或多或少的有这样一个印象:继承可以帮助我实现类的复用。

所以,很多开发人员在需要复用一些代码的时候会很自然的使用类的继承的方式,因为书上就是这么写的(老师就是这么教的)。

但是,其实这样做是不对的。长期大量的使用继承会给代码带来很高的维护成本。

前面提到复用,这里就简单介绍一下面向对象的复用技术。

复用性是面向对象技术带来的很棒的潜在好处之一。如果运用的好的话可以帮助我们节省很多开发时间,提升开发效率。但是,如果被滥用那么就可能产生很多难以维护的代码。

作为一门面向对象开发的语言,代码复用是Java引人注意的功能之一。Java代码的复用有继承,组合以及代理三种具体的表现形式。

2

继承

继承(Inheritance)是一种联结类与类的层次模型。指的是一个类(称为子类、子接口)继承另外的一个类(称为父类、父接口)的功能,并可以增加它自己的新功能的能力,继承是类与类或者接口与接口之间最常见的关系。


继承是一种is-a关系。如苹果是水果,狗是动物,哈士奇是狗。

3

组合

组合(Composition)体现的是整体与部分、拥有的关系。


组合是一种has-a的关系。如汽车有一个发动机,学校有一个老师等。

4

组合与继承的区别

首先,从类的关系确定时间点上,组合和继承是有区别的:

继承,在写代码的时候就要指名具体继承哪个类,所以,类的继承关系是在编译期就确定的。并且从基类继承来的实现是无法在运行期动态改变的,因此降低了应用的灵活性。

组合,在写代码的时候可以采用面向接口编程。所以,类的组合关系一般在运行期确定。

另外,代码复用方式上也有一定区别:

继承结构中,父类的内部细节对于子类是可见的。所以我们通常也可以说通过继承的代码复用是一种白盒式代码复用。

如果基类的实现发生改变,那么派生类的实现也将随之改变。这样就导致了子类行为的不可预知性。

组合是通过对现有的对象进行拼装(组合)产生新的、更复杂的功能。因为在对象之间,各自的内部细节是不可见的,所以我们也说通过组合的代码复用是黑盒式代码复用。

因为组合中一般都定义一个类型,所以在编译期根本不知道具体会调用哪个实现类的方法。

最后,Java中不支持多继承,而组合是没有限制的。就像一个人只能有一个父亲,但是他可以有很很多辆车。

5

优缺点对比

组合关系 继承关系

优点:不破坏封装,整体类与局部类之间松耦合,彼此相对独立

缺点:破坏封装,子类与父类之间紧密耦合,子类依赖于父类的实现,子类缺乏独立性

优点:具有较好的可扩展性

缺点:支持扩展,但是往往以增加系统结构的复杂度为代价

优点:支持动态组合。在运行时,整体对象可以选择不同类型的局部对象

缺点:不支持动态继承。在运行时,子类无法选择不同的父类

优点:整体类可以对局部类进行包装,封装局部类的接口,提供新的接口

缺点:子类不能改变父类的接口

缺点:整体类不能自动获得和局部类同样的接口

优点:子类能自动继承父类的接口

缺点:创建整体类的对象时,需要创建所有局部类的对象

优点:创建子类的对象时,无须创建父类的对象

6

为什么组合优于继承

相信很多人都知道面向对象中有一个比较重要的原则『多用组合、少用继承』或者说『组合优于继承』。

从前面的介绍已经优缺点对比中也可以看出,组合比继承更加灵活,也更有助于代码维护。其具有不破坏封装性、具有更好的可扩展性、支持动态组合、整体类可以改变局部类的行为等优点。

所以,建议在同样可行的情况下,优先使用组合而不是继承。因为组合更安全,更简单,更灵活,更高效。

注意,并不是说继承就一点用都没有了,前面说的是【在同样可行的情况下】。有一些场景还是需要使用继承的,或者是更适合使用继承。

另外,除了《阿里巴巴Java开发手册》,在很多其他资料中也有关于组合和继承的介绍和使用约束:

继承要慎用,其使用场合仅限于你确信使用该技术有效的情况。一个判断方法是,问一问自己是否需要从新类向基类进行向上转型。如果是必须的,则继承是必要的。反之则应该好好考虑是否需要继承。《Java编程思想》

快薅,当当的羊毛,晚了就没了!

快薅,当当的羊毛,晚了就没了!

快薅,当当的羊毛,晚了就没了!

- MORE | 更多精彩文章 -

  • 把 14 亿中国人都拉到一个微信群,在技术上能实现吗?

  • Linux 28 岁了,我们总结了 28 个不为人知的事实

  • 老婆背着我在豆瓣发帖:没事别嫁程序员!

  • 为什么技术团队领导者多是后台开发人员

如果你喜欢本文,

请长按二维码,关注 Hollis.

转发至朋友圈,是对我最大的支持。

好文章,我在看❤️

原创 | 为什么阿里巴巴建议开发者谨慎使用继承?相关推荐

  1. 原创 | 为什么阿里巴巴要求谨慎使用ArrayList中的subList方法

    △Hollis, 一个对Coding有着独特追求的人△ 这是Hollis的第 219 篇原创分享 作者 l Hollis 来源 l Hollis(ID:hollischuang) 集合是Java开发日 ...

  2. 为什么阿里巴巴建议集合初始化时,指定集合容量大小

    集合是Java开发日常开发中经常会使用到的.在之前的一些文章中,我们介绍过一些关于使用集合类应该注意的事项,如<为什么阿里巴巴禁止在 foreach 循环里进行元素的 remove/add 操作 ...

  3. 为什么阿里巴巴建议集合初始化时,指定集合容量大小?

    集合是Java开发日常开发中经常会使用到的.在之前的一些文章中,我们介绍过一些关于使用集合类应该注意的事项,如<为什么阿里巴巴禁止在 foreach 循环里进行元素的 remove/add 操作 ...

  4. CSDN”原力计划“在召唤:技术人请集结,用原创技术影响万千开发者

    技术深不可测.薪资难以想象.着装招人吐槽.发量让人惊叹.笑点着实密集.情商令人堪忧......在这个你我他她它通过网络紧密互联.消息实时互通的 21 世纪,人们对身处技术至高点的程序员们仍然有着以上不 ...

  5. 微软将中止支持 .NET Core 2.2,建议开发者升级至 .NET Core 3.1

    Current 版本 .NET Core 2.2 将在 12 月 23 日结束生命周期,开发者应更新到长期支持版本 .NET Core 3.1. .NET Core 2.2 于 2018 年 12 月 ...

  6. java已被弱化签名,高效Java第四十条建议:谨慎设计方法签名

    作用 有助于设计易于学习和使用的API. 如何做--谨慎地选择方法的名称 1.选择易于理解的,并且与同一个包中的其他名称风格一致的名称. 2.选择与大众认可的名称相一致的名称. 如何做--不要过于追求 ...

  7. 简直没法玩!iOS 13三指手势影响多款游戏操作,腾讯建议玩家谨慎更新

    苹果iOS 13正式版推送后,不少用户都第一时间升级了,但是iOS 13却让不少游戏用户感到苦恼,因为其三指操作竟然与<和平精英>等手游产生了冲突. 苹果在iOS 13加入了三指按住手势以 ...

  8. idea 阿里巴巴规范插件_阿里巴巴 Java 开发者手册,编码事半功倍

    4月22日,阿里巴巴发布了泰山版<Java 开发手册>,以前以为终极版就真的是终极版了,没想到还是想的太简单了,继终极版之后又发布了详尽版.华山版,这不,泰山版又来了.想想也对,行业一直在 ...

  9. 敏捷开发签名人建议开发者放弃“敏捷”

    作者|Rui Miguel Ferreira 译者|无明 Ron Jeffries 是极限编程(XP)的创始人之一,也是敏捷宣言的签名人之一,他在博客中发表了一篇文章,主张开发人员应该放弃" ...

最新文章

  1. CvBlobTrackerCC 多目标跟踪算法简析
  2. IE6下绝对定位的DIV被相对定位的DIV挡住的问题
  3. 正经聊一聊脑机接口的发展现状|脑科学开放日
  4. 公共的service接口
  5. 普通函数与函数模板的区别
  6. [Google Guava] 9-I/O
  7. html制作固定列的表格,带固定列的简单HTML表格
  8. java+输出流++空值_一文搞定Java的输入输出流等常见流
  9. Math.signbit()
  10. Java:转换列表 String 到一个字符串
  11. MySQL不应出现的开始声明_备考:2019年9月计算机二级MySQL试题
  12. HDU-1301-Jungle Roads
  13. oracle in like组合,IN操作符与LIKE子句-各种查询
  14. vs的windows应用程序上的鼠标为什么一直是加载状态?_了解 JavaScript 应用程序中的内存泄漏...
  15. Android中实现两次点击返回键退出本程序
  16. 网狐6603全部架设过程棋牌源码下载搭建教程
  17. shell 004 __小斌文档 |数组+函数
  18. 使用hibernate的@SQLDelete和@Where注解实现逻辑删除
  19. 正则表达式判断手机号码运营商
  20. 智慧幼儿园方案:AI技术如何助力幼儿园智慧建设?

热门文章

  1. 事务失败返回_什么是分布式事务以及有哪些解决方案?
  2. int指针初始化_C++:变量,指针,引用const,extern,using,typedef,decltype关键字
  3. java:数字转string 报空指针_超干货详解:kotlin(4) java转kotlin潜规则
  4. 二分法python上机实验报告_二分查找-Python刷题笔记
  5. 实现自动带密码登陆远程机执行shell命令(linux)
  6. 操作系统之内存管理:1、内存管理基础知识(指令工作原理、地址转化、程序运行过程)
  7. 计算机网络之网络层:9、BGP协议
  8. (王道408考研数据结构)第一章绪论-第二节2:算法的时间复杂度和空间复杂度
  9. Linux Socket poll
  10. python 线程的使用