为什么阿里巴巴建议开发者谨慎使用继承?

大茅坑的小石头 大茅坑的小石头     2022-11-29     581

关键词:

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

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

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

为什么阿里巴巴建议开发者谨慎使用继承?_复用

 

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

面向对象的复用技术

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

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

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

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

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

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

继承

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

为什么阿里巴巴建议开发者谨慎使用继承?_代码复用_02

 

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

组合

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

为什么阿里巴巴建议开发者谨慎使用继承?_java_03

 

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

组合与继承的区别

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

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

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

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

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

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

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

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

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

优缺点对比

为什么阿里巴巴建议开发者谨慎使用继承?_java_04

 

为什么组合优于继承

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

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

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

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

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

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

只有当子类真正是超类的子类型时,才适合用继承。换句话说,对于两个类A和B,只有当两者之间确实存在is-a关系的时候,类B才应该继续类A。《Effective Java》


为什么阿里巴巴要求程序员谨慎修改serialversionuid字段的值(代码片段)

为什么阿里巴巴要求程序员谨慎修改serialVersionUID字段的值收录于话题#解读阿里开发手册15个序列化是一种对象持久化的手段。普遍应用在网络传输、RMI等场景中。类通过实现java.io.Serializable接口以启用其序列化功能。在我的博客... 查看详情

为什么建议使用你localdatetime,而不是date?

...发过程中经常遇到时间处理,但是你真的用对了吗,理解阿里巴巴开发手册中禁用static修饰SimpleDateFormat吗通过阅读本篇文章你将了解到:为什么需要LocalDate、LocalTime、LocalDateTime【java8新提供的类】java8新的时间API的使用方式,包... 查看详情

建议4:谨慎使用运算符---谨慎使用++和--(代码片段)

...变得过于紧密,复杂和隐晦.因此,在JavaScript程序设计中不建议使用他们,从而使代码风格变得更为整洁.  ++和--运算符只能够作用域变量,数组元素或对象属性.下面用法时错误的4++;  正确的用法如下:varn=4;n++;  ++和--运算符的... 查看详情

为什么阿里巴巴不建议在for循环中使用"+"进行字符串拼接(代码片段)

为什么阿里巴巴不建议在for循环中使用"+"进行字符串拼接收录于话题#解读阿里开发手册15个字符串,是Java中最常用的一个数据类型了。关于字符串的知识,作者已经发表过几篇文章介绍过很多,如:该如何创建字符串,... 查看详情

rinetd在生产环境要谨慎使用

...现redis连不上了,提示Connectionresetbypeer排查1.排查Redis为什么会这样,首先 查看详情

阿里《java开发手册》为什么建议设置hashmap的初始容量,设置多少合适(代码片段)

 目录要设置HashMap的初始化容量HashMap初始化容量设置多少合适 集合是Java开发日常开发中经常会使用到的,而作为一种典型的K-V结构的数据结构,HashMap对于Java开发者一定不陌生。关于HashMap,很多人都对他有一些基... 查看详情

枚举很好用啊!为啥阿里不建议返回值用枚举??

...了枚举,结果被其它人深深嫌弃,说不好拓展。为什么会被嫌弃呢?我们先来看看阿里开发手册关于枚举使用的建议从手册可以看出,定义和使用枚举,阿里开发手册都是支持的,但是为啥,返回值就... 查看详情

java:effectivejava学习笔记之复合优先于继承(代码片段)

...合优于继承,多用组合少用继承。同样地,在《阿里巴巴Java开发手册》中有一条规定:谨慎使用继承的方式进行扩展,优先使用组合的方式实现。1、实现继承和接口继承实现继承:当一个类扩展另一个类的时... 查看详情

枚举很好用啊!为啥阿里不建议返回值用枚举??(代码片段)

...了枚举,结果被其它人深深嫌弃,说不好拓展。为什么会被嫌弃呢?我们先来看看阿里开发手册关于枚举使用的建议从手册可以看出,定义和使用枚举,阿里开发手册都是支持的,但是为啥,返回值就... 查看详情

阿里巴巴常用的12个后端开发工具,建议你快快收藏。。

...到创新,开发人员工具随着技术的发展而不断发展。阿里巴巴集团和阿里巴巴云已通过开源发布和基于云的实施向公众提供其技术。通过在各种业务场景中的多年开发积累了这些技术。本文介绍了一些阿里巴巴开发人员工具&... 查看详情

阿里巴巴为什么不建议直接使用async注解?

来源:码农架构对于异步方法调用,从Spring3开始提供了@Async注解,该注解可以被标在方法上,以便异步地调用该方法。调用者将在调用时立即返回,方法的实际执行将提交给SpringTaskExecutor的任务中,由... 查看详情

阿里巴巴为什么不建议直接使用async注解?

相关阅读:一个90后员工猝死的全过程对于异步方法调用,从Spring3开始提供了@Async注解,该注解可以被标在方法上,以便异步地调用该方法。调用者将在调用时立即返回,方法的实际执行将提交给SpringTaskExe... 查看详情

阿里巴巴为什么不建议直接使用async注解?(代码片段)

对于异步方法调用,从Spring3开始提供了@Async注解,该注解可以被标在方法上,以便异步地调用该方法。调用者将在调用时立即返回,方法的实际执行将提交给SpringTaskExecutor的任务中,由指定的线程池中的线... 查看详情

阿里巴巴java开发手册建议创建hashmap时设置初始化容量,但是多少合适呢?

阿里巴巴Java开发手册建议创建HashMap时设置初始化容量,但是多少合适呢?收录于话题#解读阿里开发手册15个△Hollis,一个对Coding有着独特追求的人△这是Hollis的第252篇原创分享作者lHollis来源lHollis(ID:hollischuang)集合是Java开发... 查看详情

rinetd在生产环境要谨慎使用

...现redis连不上了,提示Connectionresetbypeer排查1.排查Redis为什么会这样,首先我怀疑,是否redis有问题首先我查看了redis的负载情况,使用INFO命令#Clientsconnected_clients:192client_longest_output_list:0client_biggest_input_buf:0blocked_clie... 查看详情

建议4:谨慎使用运算符---警惕运算符得副作用(代码片段)

  在JavaScript中有一些运算符能够该百年运算数自身的值,如赋值,递增,递减运算等.  例如:vara;a=1;a=(a++)+(++a)-(a++)-(++a);console.log(a)//-4  如果直观地去判断,会错误地认为返回值是0,实际上变量a在参与运算得过程中,变量a得值不... 查看详情

到底为什么不建议使用select*?(代码片段)

来源:蝉沐风(ID:chanmufeng1994)Hollis的新书限时折扣中,一本深入讲解Java基础的干货笔记!“不要使用SELECT*”几乎已经成为了使用MySQL的一条金科玉律,就连《阿里Java开发手册》也明确表示不得使用*... 查看详情

到底为什么不建议使用select*?(代码片段)

来源:蝉沐风(ID:chanmufeng1994)Hollis的新书限时折扣中,一本深入讲解Java基础的干货笔记!“不要使用SELECT*”几乎已经成为了使用MySQL的一条金科玉律,就连《阿里Java开发手册》也明确表示不得使用*... 查看详情