第二章 创建和销毁对象

何时以及如何创建对象,何时以及如何避免创建对象,如何确保他们能够适时地销毁,以及如何管理对象销毁之前必须进行的各种清理动作。

1 考虑用静态工厂方法代替构造器

一般在某处获取一个类的实例最常用的方法是提供一个共有的构造器,还有一种方法,就是提供一个共有的静态工厂(static factory method),他只是一个返回类的实例的静态方法。

例:

 public static Boolean valueOf(boolean b){return b ? Boolean.TRUE:Boolean.FALSE;} 

注意,静态工厂方法与设计模式中的工厂方法模式不同。类可以通过静态工厂方法来提供给它的客户端,而不是通过构造器,提供静态工厂方法而不是公有的构造器,这样做具有几大优势:

  • 静态工厂方法,它们有名称
    例如构造器BIgInteger(int,int,Random)返回的BigInteter可能为素数,如果用名为BigInteger.probablePrime的静态工厂方法来表示,显然更为清楚。
  • 不必在每次调用它们的时候都创建一个新的对象

    这使得不可变类可以使用预先构建好的实例,或者将构建好的实例缓存起来,进行重复利用,从而避免常见不必要的重复对象,因为程序经常请求创建相同的对象,那么创建对象的代价会很高。Boolean.valueOf(boolean)方法说明了这项技术。静态工厂方法也经常用于实现单例模式。
  • 它们可以返回原返回类型的任何子类型的对象

灵活的静态工厂方法构成了服务提供者框架(Service Provider FrameWork)的基础,例如JDBC。

对于JDBC,Connection就是它的服务接口,DriverManager.registerDriver是提供者注册API,DriverManager.getConnection是服务访问API,Driver就是服务提供者接口。

例,下列简单的实现包含了一个服务提供者接口和一个默认提供者:

/*** Created by Newtrek on 2017/10/31.* 服务提供者接口*/
public interface Provider {
//    提供服务实例,用服务接口返回Service newService();
}
/*** Created by Newtrek on 2017/10/31.* 服务接口*/
public interface Service {// TODO: 2017/10/31  服务特有的方法 写在这儿
}

/*** Created by Newtrek on 2017/10/31.*/
public class Services {
//    构造保护private Services(){}
//    provider映射表,保存注册的Providerprivate static final Map<String ,Provider> providers=new ConcurrentHashMap<>();
//    默认provider的名字public static final String DEFAULT_PROVIDER_NAME="<def>";
//    注册默认的Providerpublic static void registerDefaultProvider(Provider p){registerProvider(DEFAULT_PROVIDER_NAME,p);}
//    注册providerpublic static void registerProvider(String name,Provider p){providers.put(name,p);}/*** 静态工厂方法返回Service实例*/public static Service newInstance(){return newInstance(DEFAULT_PROVIDER_NAME);}public static Service newInstance(String name){Provider p = providers.get(name);if (p==null){throw new IllegalArgumentException("No provider registered with name:"+name);}return p.newService();}
}
  • 在创建参数化类型实例的时候,它们是代码变得更加简洁

例:假设HashMap提供了这个静态工厂

 public static <K,V> HashMap<K,V> newInstance(){return new HashMap<K,V>();} 

那么就可以用下面简洁的代码获取实例了。

 Map<String,List<String>> m=HashMap.newInstance(); 

把这些方法放在自己的工具类中是很实用的。不过现在java7,java8已经实现了HashMap构造的类型参数推测

缺点

  • 类如果不含公有的或者受保护的构造器,就不能被子类化
  • 它们与其他的静态方法实际上没有任何区别

    • 在API文档中,它们没有像构造器那样在API文档中明确标识出来,因为,对于提供了静态工厂方法二不是构造器的类来说,要想查明如何实例化一个类,这是非常困难的。可以通过在类或者接口注释中关注静态工厂,并遵守标准的命名习惯,可以弥补这一劣势。下面是静态工厂方法的一些惯用名称。

      • valueOf:实际上是类型转换方法。
      • of:valueOf的一种更为简洁的代替
      • getInstance:返回的实例是通过方法的参数来描述的,但是不能够说与参数具有同样的值。对于Singleton来说,该方法没有参数,并返回唯一的实例。
      • newInstance:像getInstance一样,但newInstance能够确保返回的每个实例都与所有其它实例不同。
      • getType:像getInstance一样,但是在工厂方法处于不同的类中的时候使用。Type表示工厂方法所返回的对象类型。
      • newType:像newInstance一样,但是在工厂方法处于不同的类中的时候使用,Type表示工厂方法所返回的对象类型。

简而言之,静态工厂方法和公有构造器都各有用处,我们需要理解他们各自的长处。静态工厂通常更加合适,因此切忌第一反应就是提供公有的构造器,而不先考虑静态工厂。

2.遇到多个构造器参数时要考虑用构建器

这个就是Builder设计模式

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

这个是单例模式的注意事项,选择最好的单例模式实现

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

有时候需要编写一些只包含静态方法和静态域的类,这些类的名声很不好,因为有些人在面向对象的语言中滥用这样的类来编写过程化的程序,尽管如此,他们也确实有它们的好处,比如常见的工具类java.lang.Math等,都是这样。方正这样不可以实例化的类,最好把他的构造器设置为私有。

例如:

public class UtilityClass{private UtilityClass(){throw new AssertionError();}
}

5.避免创建不必要的对象

一般来说,最好能重用对象而不是再每次需要的时候就创建一个相同功能的新对象,如果对象是不可变的,他就始终可以被重用。
简单的例子:字符串

String s = new String("stringtest");//不要这样做,因为该语句每次执行的时候都会创建一个新的String实例,没必要
// 改进后的版本,这个版本只用了一个String实例,而不是每次执行的时候都创建一个新的实例。对于再同一台虚拟机中运行的代码,只要它们包含相同的字符串自字面常量,该对象就可以被重用。
String s = "stringtest";

对于同时提供了静态方法和构造器方法的不可变类,通常可以使用静态工厂方法而不是构造器,以避免创建不必要的对象。

优先使用基本类型,而不是装箱基本类型,当心无意识的自动装箱。

也不要错误地认为本条目暗示着“创建对象的代价非常昂贵,我们应该尽可能地避免创建对象”,相反,由于小对象地构造器制作很少量地显示工作,所以,小对象地创建和回收动作是非常廉价地。

通过维护自己的对象池来避免创建对象比不是一种好的做法,除非池中的对象是非常重量级的,一般数据库连接池常用。

6 消除过期的对象引用

不要以为Java有垃圾回收机制,能自动管理内存,自动回收垃圾,就可以不管了,其实不然。
内存泄漏的例子


public class Stack {private Object[] elements;private int size = 0;private static final int DEFAULT_CAPACITY = 16;public Stack(){elements = new Object[DEFAULT_CAPACITY];}public void push(Object object){ensureCapacity();elements[size++] = object;}public Object pup(){if (size == 0){throw  new EmptyStackException();}return elements[size--];}private void ensureCapacity(){if (elements.length == size){elements = Arrays.copyOf(elements,size*2+1);}}}

这段程序并没有明显的错误,如果是栈先是增长,然后再收缩,那么,从栈中弹出来的对象将不会被当作垃圾回收,因为里面的数组里引用着它,栈内部维护着这些对象的过期引用,过期引用就是指永远也不会再被解除的引用。

这类问题的修复方法很简单:一旦对象引用已经过期,只需清空这些应用即可。

没必要对于每一个对象引用,一旦程序不再用到它,就把它清空。清空对象引用应该是一种例外,而不是一种规范行为,消除过期引用最好的办法是让包含该对象的变量结束其生命周期。一般而言,只要类是自己管理内存,程序员就应该警惕内存泄漏问题,一旦元素被释放掉,则该元素中包含的任何对象引用都应该被清空。

内存泄漏的另一个常见来源是缓存,一旦你把对象放在缓存中,他就很容易被遗忘掉,从而使得它不再有用之后很长一段时间内仍然留在缓存中。可以用WeakHashMap

内存泄漏的第三个常见来源是监听器和其他回掉,一般都要取消注册,或者用弱引用

内存泄漏通常不会表现成明显的失败,所以他们可以再一个系统中存在很多年,往往通过仔细检查代码,借助于Heap刨析工具才能发现内存泄漏问题。

7 避免使用终结方法

终结方法(finalizer)通常是不可预测的,也是很危险的,一般情况下是不必要的。

C++的析构器是回收一个对象所占用资源的常规方法,是构造器所必须的对应物,也可以用来回收其他的非内存资源,而在Java中,一般用try-finally块来完成类似的工作

《Effective Java》学习笔记 第二章 创建和销毁对象相关推荐

  1. Effective Java读书笔记---二、创建和销毁对象

    二.创建和销毁对象 何时以及如何创建对象, 何时以及如何避免创建对象, 如何确保它们能够适时地销毁, 如何管理对象销毁之前必须进行的各种清理动作 1.用静态工厂方法代替构造器 优势: 它们有名称 不必 ...

  2. Effective Java读书笔记三:创建和销毁对象

    第1条:考虑用静态工厂方法代替构造器 对于类而言,为了让客服端获得它的一个实例最常用的的一个方法就是提供一个公有的构造器.还有一种方法,类可以提供一个公有的静态工厂方法(static factory ...

  3. Effictive Java学习笔记1:创建和销毁对象

    建议1:考虑用静态工厂方法代替构造器 理由:1)静态方法有名字啊,更容易懂和理解.构造方法重载容易让人混淆,并不是好主意 2)静态工厂方法可以不必每次调用时都创建一个新对象,而公共构造函数每次调用都会 ...

  4. [Effective Java]第二章 创建和销毁对象

    第一章      前言 略... 第二章      创建和销毁对象 1.            考虑用静态工厂方法代替构造器 创建对象方法:一是最常用的公有构造器,二是静态工厂方法.下面是一个Bool ...

  5. 《Effective Java》 第一讲:创建和销毁对象

    前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家.点击跳转到教程. 一.用静态工厂方法代替构造器 用静态工厂的优点 : 1. 方法有名字,更好理解. 2.不必每次调用的 ...

  6. Java 学习笔记 ------第二章 从JDK到IDE

    本章学习目标: 了解与设定PATH 了解与指定CLASSPATH 了解与指定SOURCEPATH 使用package与import管理类别 初步认识JDK与IDE的对应关系 一.第一个Java程序 工 ...

  7. 程序设计与算法三~C++面向对象程序设计~北大郭炜MOOC学习笔记~第二章:类和对象初步(新标准C++程序设计)

    以下内容为笔者手打,望读者珍惜,如有转载还请注明. chapter2:类和对象初步 数据结构+算法=程序 $2.1结构化程序设计的不足     结构化程序设计也称面向过程的程序设计,过程是用函数实现的 ...

  8. 《Go语言圣经》学习笔记 第二章 程序结构

    Go语言圣经学习笔记 第二章 程序结构 目录 命名 声明 变量 赋值 类型 包和文件 作用域 注:学习<Go语言圣经>笔记,PDF点击下载,建议看书. Go语言小白学习笔记,几乎是书上的内 ...

  9. [go学习笔记.第二章] 2.go语言的开发工具以及安装和配置SDK

    一.工具介绍: 1.Visual Studio Code 一个运行于Mac,Windows,和linux上的,默认提供Go语言的语法高亮的IED,可以安装Go语言插件,还可以支持智能提示,编译运行等功 ...

最新文章

  1. java中遍历Map对象的四种方式
  2. 最通俗易懂的面向对象著作
  3. select下拉框下拉跳转代码
  4. 使用原生 Java 玩转验证码【含 DATA-URIS 介绍】
  5. python创建数据库表空间_7.自动化监控多个Oracle表空间
  6. [vue] 在使用计算属性的时,函数名和data数据源中的数据可以同名吗?
  7. LintCode 1862. 给树浇水的时间(图的遍历)
  8. Bundle Adjustment简述(转载)
  9. 第五章---引入复制后的数据库架构
  10. 动态链接库实现COM(COM技术内幕笔记之二)
  11. vue router-link子级返回父级页面
  12. 关于js的引用类型和基本类型
  13. [C++] vector 初始化
  14. unigui作中间件使用
  15. oracle rman备份和恢复数据库,Oracle rman备份和还原恢复数据库
  16. 定积分证明题例题_数列极限求法十五种(25个例题+推文送给微积分和数学分析同学)...
  17. 网易云邮箱验证码注册及修改密码
  18. 合肥有哪些不错的 IT 公司?
  19. 计算机表格设置宽度,word表格大小调整固定单元格大小设置——想象力电脑应用...
  20. vw/vh:移动适配之vw/vh(使用方法)

热门文章

  1. 学习笔记Hive(一)—— Hive简介
  2. 学习笔记(十八)——MongoDB(CRUD)与Python交互
  3. @Autowired和@Resouce的区别
  4. python 状态机教程_python 实用工具状态机transitions
  5. js 定时网页点击_JS的小乐趣:轻松完成打地鼠游戏
  6. “http://127.0.0.1:8888' has been blocked by CORS policy: No 'Access-Control-Allow-Origin' header”
  7. ViewModel优雅的弹加载窗和获取Context
  8. android view setx,Android的setX()和setY()表现不可思议
  9. 半潜式平台及其动力定位系统
  10. c语言 文件 long double 读取,读取*.wav音频文件