EFFECTIVE JAVA 类和接口
2024-09-06 16:42:49
第十六条:复合优先于继承
//这是一个不好的类---执行的结果 addCount = 4(addAll的实现依赖于HashSet的add方法,InstrumentHashSet方法重写了add方法有执行了addCount++)
public class InstrumentHashSet<E> extends HashSet<E> {
private int addCount = 0 ; @Override
public boolean add(E e) { System.out.println("子类添加 ");
addCount++;
return super.add(e);
} @Override
public boolean addAll(Collection<? extends E> c) {
addCount+= c.size();
System.out.println("添加所有 ");
return super.addAll(c);
} public static void main(String[] args) {
InstrumentHashSet s = new InstrumentHashSet<String>();
s.addAll(Arrays.asList("1","2"));
System.out.println(s.addCount);
}
}
//wrapper class --- use composition in place of inheritance 因为每一个InstrumentedSet 实例都把另一个Set实例包装起来了,所以InstrumentedSet类被称为包装类(这正是Decorator模式)---注意这不是委托(delegation)除非包装对象把自身传递给被包装对象
public class InstrumentedSet<E> extends ForwardingSet<E> {
private int addCount = 0 ;
/**
* 描述: 构造方法
* @param s
*/
public InstrumentedSet(Set<E> s) {
super(s);
}
@Override
public boolean add(E e) { System.out.println("我执行了InstrumentedSet 的add");
addCount++;
return super.add(e);
}
@Override
public boolean addAll(Collection<? extends E> c) {
System.out.println("执行 addAll c.size= " + c.size() +" count = "+ addCount); addCount += c.size();
return super.addAll(c);
} public int getAddcount(){
return addCount;
}
public static void main(String[] args) {
HashSet<Integer> s = new HashSet<Integer>();
s.add(1);
s.add(2); HashSet<Integer> s1 = new HashSet<Integer>();
s1.add(3);
s1.add(4);
InstrumentedSet is = new InstrumentedSet<Integer>(s);
// is.add(3);
is.addAll(s1);
System.out.println(is.getAddcount()); } }
//forwarding class 里面的转发方法称为转发方法(forwarding method) 这样得到的类非常的稳固,它不依赖于现有类的实现细节,即使现有的类添加了新的方法,也不会影响新的类
public class ForwardingSet<E> implements Set<E> {
private Set<E> s ; /**
* 描述: 构造方法
*/
public ForwardingSet(Set<E> s) {
System.out.println("初始化s,类型:"+s.getClass());
this.s = s;
}
@Override
public int size() {
return s.size();
}
@Override
public boolean isEmpty() {
return s.isEmpty();
}
@Override
public boolean contains(Object o) {
return s.contains(o);
}
@Override
public Iterator<E> iterator() {
return s.iterator();
}
@Override
public Object[] toArray() {
return s.toArray();
}
@Override
public <T> T[] toArray(T[] a) {
return s.toArray(a);
}
@Override
public boolean add(E e) {
System.out.println("我执行了ForwardingSet 的add");
return s.add(e);
}
@Override
public boolean remove(Object o) {
return s.remove(o);
}
@Override
public boolean containsAll(Collection<?> c) {
return s.containsAll(c);
}
@Override
public boolean addAll(Collection<? extends E> c) {
System.out.println("我执行了addAll");
return s.addAll(c);
}
@Override
public boolean retainAll(Collection<?> c) {
return s.retainAll(c);
}
@Override
public boolean removeAll(Collection<?> c) {
return s.removeAll(c);
}
@Override
public void clear() {
s.clear();
} }
测试中的addAll方法实际上走的是HashSet的add方法
最新文章
- 使用F#开发ASP.NET Core应用程序
- 基于dubbo的分布式项目实例应用
- 关于vue.js中列表渲染练习
- Struts2 Action中动态方法调用、通配符的使用
- 期望DP
- SSAS:概念梳理
- 【WPF】WPF中调用Winform
- z-index兼容问题:关于ie6/7下的z-index
- Android实例-解决虚拟键盘遮挡问题(XE8+小米2)
- mysql datetime 排序
- ios 好去处
- 在Visual Studio中使用AStyle
- 2015第14周日WebSocket
- ADO读取EXCEL
- 如何将ASP.NET-WebApi发布到IIS6.0上(转)
- Tornado 网站demo 一
- es6下 vue实例属性template不能使用
- C++类的内存结构
- Docker 技巧:删除 Docker 所有镜像
- imperva 默认策略添加例外
热门文章
- nginx+lua_module安装
- SSM整合过程中出现的问题
- JavaScript 开发者的 10 款必备工具
- Flask插件系列之flask_celery
- POJ 2923 【01背包+状态压缩/状压DP】
- java中split任意数量的空白字符
- NMAP输出结果中CPE的含义
- luogu P3818 小A和uim之大逃离 II
- jsp笔记1(基本原理与语法)
- 【spring boot】14.spring boot集成mybatis,注解方式OR映射文件方式AND pagehelper分页插件【Mybatis】pagehelper分页插件分页查询无效解决方法