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

简介: 第二章 创建和销毁对象何时以及如何创建对象,何时以及如何避免创建对象,如何确保他们能够适时地销毁,以及如何管理对象销毁之前必须进行的各种清理动作。1.考虑用静态工厂方法代替构造器一般在某处获取一个类的实例最常用的方法是提供一个共有的构造器,还有一种方法,就是提供一个共有的静态工厂(static factory method),他只是一个返回类的实例的静态方法。

第二章 创建和销毁对象

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

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。
服务提供者框架:多个服务提供者实现一个服务,系统为服务提供者的客户端提供多个实现,并把他们从多个实现总解耦出来。三个组件:服务接口,这是服务提供者实现的;提供者注册API,这是系统用来注册实现,让客户端访问他们的;服务访问API,是客户端用来获取服务的实例的。
对于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映射表,保存注册的Provider
    private static final Map<String ,Provider> providers=new ConcurrentHashMap<>();
//    默认provider的名字
    public static final String DEFAULT_PROVIDER_NAME="<def>";
//    注册默认的Provider
    public static void registerDefaultProvider(Provider p){
        registerProvider(DEFAULT_PROVIDER_NAME,p);
    }
//    注册provider
    public 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块来完成类似的工作

目录
相关文章
|
2月前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
3月前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
62 17
|
2月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
3月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
38 6
|
2月前
|
Java 数据库连接 API
Spring 框架的介绍(Java EE 学习笔记02)
Spring是一个由Rod Johnson开发的轻量级Java SE/EE一站式开源框架,旨在解决Java EE应用中的多种问题。它采用非侵入式设计,通过IoC和AOP技术简化了Java应用的开发流程,降低了组件间的耦合度,支持事务管理和多种框架的无缝集成,极大提升了开发效率和代码质量。Spring 5引入了响应式编程等新特性,进一步增强了框架的功能性和灵活性。
55 0
|
3月前
|
Oracle Java 关系型数据库
重新定义 Java 对象相等性
本文探讨了Java中的对象相等性问题,包括自反性、对称性、传递性和一致性等原则,并通过LaptopCharger类的例子展示了引用相等与内容相等的区别。文章还介绍了如何通过重写`equals`方法和使用`Comparator`接口来实现更复杂的相等度量,以满足特定的业务需求。
36 3
|
3月前
|
存储 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第9天】在Java的世界里,对象序列化是连接数据持久化与网络通信的桥梁。本文将深入探讨Java对象序列化的机制、实践方法及反序列化过程,通过代码示例揭示其背后的原理。从基础概念到高级应用,我们将一步步揭开序列化技术的神秘面纱,让读者能够掌握这一强大工具,以应对数据存储和传输的挑战。
|
3月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第3天】Java零基础教学篇,手把手实践教学!
44 1
|
3月前
|
Java 数据安全/隐私保护
java类和对象
java类和对象
29 5
|
2月前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
58 0