小伙伴们好呀, 这次分享的是 Effective Java 第三弹~ 类和接口 (15-18 小节)
其实读到这一章的时候,我已经开始烦躁了……
可能是因为这是让我从头学 Java ,从最开始的 类和接口 去探索,心里怎么都有点不服
在废了好一番功夫后,才逐渐让自己这颗 急功近利 的心稍微安静下来
这篇文章也是和我的心情差不多,开始的随意,后面的渐渐有了思考扩展,收获还是很大的,但是书还得耐心看下去~
make each class or member as inaccessible as possible.
这个就是能 private 的就弄成 private
包括属性,内部类。
private < default < protected < public
权限范围 | private | default | protected | public |
---|---|---|---|---|
同包同类 ✔ | ✔ | ✔ | ✔ | |
同包不同类 | ✔ | ✔ | ✔ | |
不同包子类 | ✔ | ✔ | ||
不同包非子类 | ✔ |
就是属性要 private,方法要 public
这一节介绍了这个 不可变类 ,比如:String,基本数据类型的包装类,BigInteger,BIgDecimal
最后一点比较难理解,这里看看 String 的例子。
揭秘
所以要避免创建过多的不可变类对象,比如:String 有 StringBuilder(线程不安全版本) 相助一样,如果要拼接很长的字符串,要用它去创建,这样就不会每一步都创建出一个 string 对象而占用太多内存了,因为 StringBuilder 内部用 char 数组缓冲区来拼接字符串
StringBuilder b = new StringBuilder("hello");
b.append(" and good");
答案是:4 个, StringBuilder 一个,两个字符串常量,还有内部的 char 数组
这里我感受最深的是这个 不可变类的深拷贝 ,以前还真没去留意 这些不可变类,以及他们怎么处理内部的可变类属性的!还有就是应对不可变类的缺点,需创建更多对象而带来的一些策略,如 StringBuilder 的 char 缓冲区 利于拼接,Integer 的 cache 缓存数组重复利用 -128 到 127 的数
技能get ✔
看题先
这里输出的 addCount 是多少?
package com.jdk8.effective.chapter4;
import java.util.*;
// Broken - Inappropriate use of inheritance! (Page 87)
public class InstrumentedHashSet<E> extends HashSet<E> {
// The number of attempted element insertions
private int addCount = 0;
public InstrumentedHashSet() {
}
public InstrumentedHashSet(int initCap, float loadFactor) {
super(initCap, loadFactor);
}
@Override public boolean add(E e) {
addCount++;
return super.add(e);
}
@Override public boolean addAll(Collection<? extends E> c) {
addCount += c.size();
return super.addAll(c);
}
public int getAddCount() {
return addCount;
}
public static void main(String[] args) {
InstrumentedHashSet<String> s = new InstrumentedHashSet<>();
s.addAll(Arrays.asList("Snap", "Crackle", "Pop"));
System.out.println(s.getAddCount());
}
}
思考后再往下滑动~
答案是 6
问题便出现在这个 addAll 方法中,他会遍历调用 add 方法,所以 count = 3+1+1+1
通过这个例子,我们也可以看出 继承的风险 ,比如
同时,作者举了 JDK 中两个写的不好的例子:Properties,Stack,表示他们应该用复合而不是继承。(继承要满足 is 关系)
接着,便引出这个 复合
package com.jdk8.effective.chapter4;
import java.util.*;
// Reusable forwarding class (Page 90)
public class ForwardingSet<E> implements Set<E> {
private final Set<E> s;
public ForwardingSet(Set<E> s) { this.s = s; }
public void clear() { s.clear(); }
public boolean contains(Object o) { return s.contains(o); }
public boolean isEmpty() { return s.isEmpty(); }
public int size() { return s.size(); }
public Iterator<E> iterator() { return s.iterator(); }
public boolean add(E e) { return s.add(e); }
public boolean remove(Object o) { return s.remove(o); }
public boolean containsAll(Collection<?> c)
{ return s.containsAll(c); }
public boolean addAll(Collection<? extends E> c)
{ return s.addAll(c); }
public boolean removeAll(Collection<?> c)
{ return s.removeAll(c); }
public boolean retainAll(Collection<?> c)
{ return s.retainAll(c); }
public Object[] toArray() { return s.toArray(); }
public <T> T[] toArray(T[] a) { return s.toArray(a); }
@Override public boolean equals(Object o)
{ return s.equals(o); }
@Override public int hashCode() { return s.hashCode(); }
@Override public String toString() { return s.toString(); }
}
package com.jdk8.effective.chapter4;
import java.util.*;
// Wrapper class - uses composition in place of inheritance (Page 90)
public class InstrumentedSet<E> extends ForwardingSet<E> {
private int addCount = 0;
public InstrumentedSet(Set<E> s) {
super(s);
}
@Override public boolean add(E e) {
addCount++;
return super.add(e);
}
@Override public boolean addAll(Collection<? extends E> c) {
addCount += c.size();
return super.addAll(c);
}
public int getAddCount() {
return addCount;
}
public static void main(String[] args) {
InstrumentedSet<String> s = new InstrumentedSet<>(new HashSet<>());
s.addAll(Arrays.asList("Snap", "Crackle", "Pop"));
System.out.println(s.getAddCount());
}
}
现在,输出结果便是 3 了。
解析:InstrumentedSet 中 +3 后,就调用 HashSet 的 addAll 方法了。我们没有重写 HashSet 中的 add 方法,自然不会出现重复计算了。
例子1 例子2 对比
图中忘记加上 field 了。
例子2 ForwardingSet 中多了 Set 字段,拿例子来说,实际功能便是 为 HashSet 增加一个 统计功能 addCount 。
答案就是 装饰者模式
这里写个小例子感受下
package com.jdk8.effective.chapter4;
import java.util.Collection;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;
/**
* @author Java4ye
* @微信公众号:Java4ye
* @GitHub https://github.com/Java4ye
* @CSDN https://blog.csdn.net/weixin_40251892
* @掘金 https://juejin.cn/user/2304992131153981
*/
public class DecorateSet<E> implements Set<E> {
private final Set<E> mySet;
public DecorateSet(Set<E> mySet) {
this.mySet = mySet;
}
@Override
public boolean add(E e) {
// 增加功能
System.out.println("hello Java4ye");
return mySet.add(e);
}
@Override
public Iterator<E> iterator() {
System.out.println("-------");
return mySet.iterator();
}
public static void main(String[] args) {
DecorateSet<String> decorateSet = new DecorateSet<>(new HashSet<>());
decorateSet.add("我是装饰者");
for (String s : decorateSet) {
System.out.println(s);
}
}
...
}
这一节看了之后,感受更多的是不要乱用继承关系 ,要遵循 is 关系,如果要扩展原来的功能,要更多的考虑 复合,如 装饰者模式
15 小节,主要复习下这个 可访问性
16 小节,就是属性要 private,方法要 public
17 小节,主要是 不可变类 的特点以及优缺点,还有 深拷贝 的应用
18 小节,告诉了我们 不要随意使用 继承 ,而要优先考虑这个 复合,如 装饰者模式 去扩展功能,提供系统的灵活性
戒骄戒躁!这次要耐心读完!
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8