使用继承可能破坏类的封装性,可能的问题包括:
1.子类必须知道父类的内部实现才能正确的进行覆盖,如以下父类:
public class Container { private List<Object> elements=new ArrayList<Object>(); public boolean add(Object e){ return elements.add(e); } public void addAll(Collection<Object> c){ for(Object e:c)add(e); } }
以下CounterContainer继承Container,并且可统计自从对象创建后所添加的元素的总数:
package com.bingo.practice.effective.four.sixteen; import java.util.Collection; public class CounterContainer extends Container { private int count; @Override public boolean add(Object e) { count++; return super.add(e); } @Override public void addAll(Collection<Object> c) { count+=c.size(); super.addAll(c); } public int getCount(){ return count; } }
以下测试将失败,尽管调用CounterContainer的addAll()只添加了3个元素,但是getCount()返回的值是6。这是因为Container的addAll()内部调用了add()添加元素,进而导致CounterContainer重复计数。
@Test public void testCount(){ Object[] eles=new String[]{"1","2","3"}; CounterContainer c=new CounterContainer(); c.addAll(Arrays.asList(eles)); Assert.assertEquals(c.getCount(), 3); }
如果要正确计数,CounterContainer的addAll()必须去掉计数运算。对于开发者来说,要正确的实现CounterContainer的计数功能,必须知道父类Container内部是如何添加元素的。因此继承破坏了类的封装
2.父类可能更改内部实现,导致子类实现的功能不正确。
如上例,如果Container的addAll()实现方式后期修改如下:
public void addAll(Collection<Object> c){ elements.addAll(c); }
那么CounterContainer也必须进行修改,否则计数不正确。这个问题可能导致一旦修改了父类,那么可能所有子类都必须进行相应修改
3.子类可能不小心覆盖了父类方法
如上例,如果CounterContainer增加了remove(),此时Container还未有remove()。但是在后期发布时Container也增加了remove(),此时可能导致CounterContainer功能不正常
可以使用Composition替代继承(即使用Wrapper,或Decoration设计模式),如下:
注:如果Container实现了功能接口,可创建一个基类实现此接口,然后所有Wrapper类继承此基类
public class ContainerWrapper { private Container container; private int count; public ContainerWrapper(Container container){ this.container=container; } public boolean add(Object e) { count++; return container.add(e); } public void addAll(Collection<Object> c) { count+=c.size(); container.addAll(c); } public int getCount(){ return count; }
使用Composition可能的问题包括:
1.回调方法问题,Container不能回调ContainerWrapper的方法
2.性能问题,需多创建Wrapper类对象(此问题影响较小)
仅在此情况下使用继承:如果每个B的确都是A的子类,那么使用继承
在此情况下仍然考虑使用Composition:如果父类的API具有很多缺点,那么可使用Composition提供新的API
相关推荐
arcgis api for javascript v4.16 v3.33, 包含了3和4两个版本的 api,欢迎大家下载
linux-4.16.10.tar.xz kernel 内核 可编译
Odin Multi Downloader v4.16
计算机组成与性能设计 4.16
spring4.16版本全部的包,官网来的,下载并使用
sip-4.16.6 软件,需要的下载~
linux-4.16.8.tar.gz 内核源码 kernel 编译内核用
ubuntu 4.16内核升级包,解决无线网卡和各种兼容问题,安装方法及无线网卡解决方式见我的博客https://blog.csdn.net/qq_28901541/article/details/86242029
linux-4.16.12.tar linux-4.16内核源码。Linux是一种开源电脑操作系统内核。它是一个用C语言写成,符合POSIX标准的类Unix操作系统。 [1] Linux最早是由芬兰 Linus Torvalds为尝试在英特尔x86架构上提供自由的类Unix...
4.16MySchoolApplicion4.16MySchoolApplicion4.16MySchoolApplicion4.16MySchoolApplicion4.16MySchoolApplicion4.16MySchoolApplicion4.16MySchoolApplicion4.16MySchoolApplicion4.16MySchoolApplicion
如题,Drive Fitness Test v4.16 绿色硬盘版,直接解压缩到 C 盘,重启电脑即可使用。 制作这个版本的原因是:官网上下载的 DFT 需要软驱或者光驱才能使用,不是很方便。 Drive Fitness Test v4.16 绿色硬盘版 ...
vs2012连接 sqlite 试用版 4.16版本
三菱MX-Component4.16S安装包链接,支持win7,win8;语言VB,C++,C#;三菱MX-Component4.16S安装包链接;三菱MX-Component4.16S安装包链接;
CPS4.16
avrstudio4.16的汉化包,下载解压缩之后替换原有文件即可,非常方便
最新版windows eclipse-SDK-4.16-win32-x86_64.zip
C++并发编程实战代码
tampermonkey-4.16.6160-an+fx.xpi
Tampermonkey 油猴插件是一款免费的浏览器扩展和最为流行的用户脚本管理器,它适用于 Chrome, Microsoft Edge, Safari, Opera Next, 和 Firefox。虽然有些受支持的浏览器拥有原生的用户脚本支持,但 Tampermonkey...
最新 三菱PLC通讯软件 MX Component 4.16S 兼容所有三菱PLC, C# 与 三菱PLC通讯软件,文件大小400M,内涵下载连接及提取码