目录
第1条:考虑使用静态工厂方法代替构造器
第2条:遇到多个构造器参数时考虑用构建器
第3条:用私有构造器或者枚举类型强化Singleton属性
第4条:通过私有构造器强化不可实例化的能力
第5条:避免创建不必要的对象
第6条:消除过期的对象引用
第7条:避免使用终结方法

第2章 创建和销毁对象

第1条:考虑使用静态工厂方法代替构造器

通过使用静态工厂方法而不是使用构造器来创建类。

举例:Boolean.valueOf(boolean)方法,将boolean转换为Boolean对象引用。

有以下优势:

1)静态工厂方法有名称,可以表示方法的意思

2)不必在每次调用的时候都创建新对象(对象缓存能力)。

不可变类可预先构建实例,缓存起来重复使用。

3)可以返回原返回类型的任何子类型的对象

4)在创建参数化类型的时候代码更简洁。

//我们平时创建list
List<String> list = new Arraylist<>();
//使用google 工具类
List<String> list = Lists.newArrayList();
静态工厂方法常用名称:

valueOf、of、getInstance、newInstance、getType、newType(如newArrayList)

第2条:遇到多个构造器参数时考虑用构建器

当有很多个构造参数,且有几个参数是可选的,考虑使用Builder

public class NutritionFacts {
/**
必填参数
*/
private final int servingSize;
private final int servings;
/**
可选参数
*/
private final int fat;
private final int sodium;
public static class Builder {
private final int servingSize;
private final int servings;
private int fat = 0;
private int sodium = 0;
public Builder(int servingSize, int servings) {
this.servingSize = servingSize;
this.servings = servings;
}
public Builder fat(int fat) {
this.fat = fat;
return this;
}
public Builder sodium(int sodium) {
this.sodium = sodium;
return this;
}
public P build() {
return new P(this);
}
}
private NutritionFacts(Builder builder) {
this.servingSize = builder.servingSize;
this.servings = builder.servings;
this.fat = builder.fat;
this.modium = builder.modium;
}
}

客户端代码

NutritionFacts p = new NutritionFacts.Builder(200, 8).fat(100).sodium(35).build();

更高级使用:

public interface Builder<T> {
public T build();
}
public static class NutritionFacts.Builder implements Builder<P>

这样可以将Builder<NutritionFacts>传给方法,并结合抽象工厂创建NutritionFacts实例。

第3条:用私有构造器或者枚举类型强化Singleton属性

举例1:使用私有构造器

public class Singleton {
private static final Singleton INSTANCE = new Singleton();
private Singleton() {} //私有构造器
//other code
}

为了保证Singleton类是可序列化的

1)声明加上implents Serializable

2) 所有实例域都是transient

3)提供一个readResolve方法

private Object readResolve() {
return INSTANCE;
}

举例2:使用枚举

public enum Singleton {
INSTANCE;
}

第4条:通过私有构造器强化不可实例化的能力

主要用于在写工具类的时候。

public class XXXUtils {

private XXXXUtils() {
}
//other code
}

第5条:避免创建不必要的对象

当你应该重用现有对象的时候,不要创建新的对象。

举例1:

m方法会被频繁调用时,会创建n多的String实例。

public String m() {
String s = new String("str");
//other code
}

改进后,m方法被频繁调用,但是s会被复用。

public String m() {
String s = "str";
}

举例2:

求所有Integer的和,因声明为Long sum,而不是long sum,程序将创建约2的31次方个Long实例。

public static void main(String[] args) {
Long sum = 0L;
for (long i = 0; i < Integer.MAX_VALUE; i++)
{
sum += i; //这里每次都会创建一个Long实例。要当心无意识的自动装箱。
}
System.out.println(sum);
}

第6条:消除过期的对象引用

主要讲了内存泄露问题。

举例1:类自己管理内存,易导致内存泄露。

下面是简单的栈实现,程序每次测试都会通过,但是有个隐藏问题——”内存泄露“。

栈先增长,再收缩,栈中弹出的对象不会被当做垃圾回收,即使使用栈的程序不再引用这些对象。

public class Stack {
private Object[] elements;
private int size;
private static final int DEFAULT_INIT_CAPACITY = 16;
public Stack() {
elements = new Object[DEFAULT_INIT_CAPACITY];
}
public void push(Object e) {
ensureCapacity();
elements[size++] = e;
}
public Object pop() {
if (size == 0) {
throw new EmptyStackException();
}
Object result = elements[--size];
//解决内存泄露的方法
//elements[size] = null; 清空过期引用
return result;
}
}

为何会出现内部泄露?

Stack通过数组保存数组元素,相当于自己管理内存。只要是类自己管理内存,就应该警惕内存泄露问题。

举例2:内存泄露的另一个常见来源是缓存

缓存内存泄露:把对象引用放到缓存中,容易被遗忘,不再有用之后仍留在缓存中。

情形1:只要在缓存外有对某个项的键的引用,该项就有意义。

解决方法:使用WeakHashMap。

(记住只有当缓存项生命周期由该键的外部引用而不是值决定时,WeakHashMap才有意义)

情形2:缓存项生命周期是否有意义,不是很容易确定

解决方法:后台线程定期清理 or 缓存添加新条目时顺便清理(LinkedHashMap.removeEldestEntry()方法)

情形3:更复杂的缓存

解决方法:使用java.lang.ref

举例3:内存泄露的第三个常见来源是监听器和其他回调

比如注册回调,但是没有显式地取消回调。解决方法:保存它们的弱引用(weak ref),如只将它们保存为WeakHashMap的键。

第7条:避免使用终结方法

终结方法就是finalize()方法。

Java语言规范不保证终结方法会被及时地执行,更不保证一定会被执行。

System.gc()增加了终结方法被执行的机会,但不保证一定会被执行。

《Effective Java 2nd》第2章 创建和销毁对象的更多相关文章

  1. [Effective Java 读书笔记] 第二章 创建和销毁对象 第一条

    第二章  创建和销毁对象 第一条 使用静态工厂方法替代构造器,原因: 静态工厂方法可以有不同的名字,也就是说,构造器只能通过参数的不同来区分不同的目的,静态工厂在名字上就能表达不同的目的 静态工厂方法 ...

  2. [Effective Java读书笔记] 第二章 创建和销毁对象(1~7)

    我的技术博客经常被流氓网站恶意爬取转载.请移步原文:http://www.cnblogs.com/hamhog/p/3537576.html,享受整齐的排版.有效的链接.正确的代码缩进.更好的阅读体验 ...

  3. 《Effective Java》第1章 创建和销毁对象

    第1条 用静态工厂方法代替构造器 这个静态工厂,与设计模式中的静态工厂不同,这里的静态工厂方法,替换为“静态方法”比较好理解,主要就是建议编写静态方法来创建对象. 使用静态方法的好处: 1.静态方法有 ...

  4. [Effective Java 读书笔记] 第二章 创建和销毁对象 第六-七条

    第六条 消除过期引用 JAVA中依然会有 memory leak的,比如一个栈先增长再收缩,那么从栈中弹出的对象是不会被当做垃圾回收的,即时使用栈的程序不再引用这些对象.这是因为栈的内部维护着对这些对 ...

  5. [Effective Java 读书笔记] 第二章 创建和销毁对象 第五条

    第五条 避免创建不必要的对象 书中一开始举例: String s = new String("stringette"); // don't do this //应该使用下面,只会创 ...

  6. [Effective Java 读书笔记] 第二章 创建和销毁对象 第三 四条

    第三条 用私有构造器或者枚举类型强化singleton属性 singleton指只能被实例化一次的类,即将构造器设置为私有,使用公有静态成员来实例化,且只实例化一次对象 第四条 通过私有构造器强化不可 ...

  7. [Effective Java 读书笔记] 第二章 创建和销毁对象 第二条

    第二条 遇到多个构造器参数时,可以考虑用构建器 当遇到有多个构造器参数时,常见的是用重叠构造器,即: public class TestClass{ public TestClass(int para ...

  8. 《Effective Java》第2章 创建和销毁对象

    第2条:遇到多个构造器参数时要考虑用构建器 与构造器相比,builder的微略优势在于,builder可以有多个可变(varargs)参数.构造器就像方法一样,只能有一个可变参数.因为builder利 ...

  9. 《Effective Java》读书笔记 - 2.创建和销毁对象

    Chapter 2 Creating and Destroying Objects item 1:Consider static factory methods instead of construc ...

随机推荐

  1. MQTT的学习研究(十三) IBM MQTTV3 简单发布订阅实例

    使用IBM MQTTv3实现相关的发布订阅功能 MQTTv3的发布消息的实现: package com.etrip.mqttv3; import com.ibm.micro.client.mqttv3 ...

  2. python的三个函数(eval、exec、complie)和python版RMI

    一.python的三个函数: 1.eval函数: 之前已经讲过了这个函数,该函数也类似于php的eval,例如下边这个例子 eval("os.system('id')") 但是有个 ...

  3. Unity3D之Unity3D 4.3.0 破解方法

    Dear  All 破解有风险,破解不尊重知识产权,如果有涉及请删除或者联系我……以下呢 是我这几天捣鼓的4.3.0版本 供学习!请大家支持正版! 1.下载最新版本 我是在Unity官网下载的最新版本 ...

  4. 删除Linux的用户

    vipw : root@ www.linuxidc.com :/home# vipw 找到你之前创建的用户,用dd删除那行(记得保存:wq or :x)..之后. root@ www.linuxidc ...

  5. 170531、FormData 对象的使用

    通过FormData对象可以组装一组用 XMLHttpRequest发送请求的键/值对.它可以更灵活方便的发送表单数据,因为可以独立于表单使用.如果你把表单的编码类型设置为multipart/form ...

  6. 由于dns服务为启动导致的GI集群启动故障

    1.物业由于突然断电导致grid集群重新启动后rac数据库无法正常启动,对集群进行检查,结果如下,发现其中有4个数据库状态为instance shutdown.[root@node1 ~]# su - ...

  7. 磁盘 I/O 性能监控的指标

    指标 1:每秒 I/O 数(IOPS 或 tps) 对于磁盘来说,一次磁盘的连续读或者连续写称为一次磁盘 I/O, 磁盘的 IOPS 就是每秒磁盘连续读次数和连续写次数之和.当传输小块不连续数据时,该 ...

  8. With all Java versions it is strongly recommended to not use experimental -XX JVM options.

    https://lucene.apache.org/solr/7_6_0//SYSTEM_REQUIREMENTS.html System Requirements Apache Solr runs ...

  9. ubuntu打开txt乱码

    因为不支持中文 输入命令: iconv -f gbk -t utf8 filename.txt > filename.txt.utf8

  10. B. Factory Repairs---cf627B(线段树)

    题目链接:http://codeforces.com/problemset/problem/627/B 题意:有一个工厂生产零件,但是机器是不正常的,需要维修,维修时间是 k 天,在维修期间不能生产, ...