本文参考

本篇文章参考自《Effective Java》第三版第六条"Avoid creating unnecessary objects"

avoid creating unnecessary objects by using static factory methods in preference to constructors on immutable classes that provide both

若一个类同时提供了静态工厂方法和构造方法,尽量使用静态工厂方法,因为静态工厂方法"they are not required to create a new object each time they're invoked"

这在第一条"用静态工厂方法代替构造器"中已有提及,例如boolean的装箱类Boolean,它的valueOf()方法和构造器都能返回一个指定布尔值的实例,但是构造器会新创建一个对象,而valueOf()方法会返回类内已有的一个静态不可变实例

public static final Boolean FALSE = new Boolean(false);

注意,不要因此而时时避免创建对象,现代的JVM都能够很好地应对小对象的创建,并且创建额外的对象能够提高程序的清晰性和简化部分逻辑

On the contrary, the creation and reclamation of small objects whose constructors do little explicit work is cheap, especially on modern JVM implementations. Creating additional objects to enhance the clarity, simplicity, or power of a program is generally a good thing

some object creations are much more expensive than others.If you're going to need such an "expensive object" repeatedly, it may be advisable to cache it for reuse

这往往不太容易发现,在某些类的方法被调用时,可能会产生另外一个中间的类实例,例如String.match()方法被调用时,会生成一个中间的Pattern类实例,最好将Pattern类实例缓存(如作为静态不可变字段),直接用Pattern类实例匹配正则表达式

as is often the case with lazy initialization, it would complicate the implementation with no measurable performance improvement

懒加载模式的代码实现往往会因为多线程访问而变得复杂,可能不会带来明显的性能提升,可以使用饿汉模式,在类加载时便对静态字段进行构建

prefer primitives to boxed primitives, and watch out for unintentional autoboxing

尽量使用基类型进行运算,因为装箱类型可能会有中间对象的创建和销毁,例如Long类型的加法运算

private static long sum() {
    Long sum = 0L;
    for (long i = 0; i <= Integer.MAX_VALUE; i++)
        sum += i;
    return sum;
}

程序在for循环内构造了多余的2^31个多余的Long实例

avoiding object creation by maintaining your own object pool is a bad idea unless the objects in the pool are extremely heavyweight

只有在对象的创建十分昂贵时,才进行池化,例如数据库连接池,现代的JVM的垃圾回收器性能很容易就会超过轻量级对象池的性能

Generally speaking, however, maintaining your own object pools clutters your code, increases memory footprint, and harms performance. Modern JVM implementations have highly optimized garbage collectors that easily outperform such object pools on lightweight objects

consider the case of adapters, also known as views

视图是一种特殊的情况,例如Map接口的keySet方法返回该Map对象的Set 视图

Returns a Set view of the keys contained in this map. The set is backed by the map, so changes to the map are reflected in the set, and vice-versa.

@Test
public void keySetTest() {

  Map<String, String> map = new HashMap<>(3);

  map.put("first key", "john");

  map.put("second key", "Sara");

  map.put("third key", "Mike");

  Set<String> set = map.keySet();

  for (String key : set) {

    System.out.println(key);
  }

  System.out.println("after remove the second key");

  map.remove("second key");

  for (String key : set) {

    System.out.println(key);
  }
}

Effective Java —— 避免创建不必要的对象的更多相关文章

  1. Java中创建(实例化)对象的五种方式

    Java中创建(实例化)对象的五种方式1.用new语句创建对象,这是最常见的创建对象的方法. 2.通过工厂方法返回对象,如:String str = String.valueOf(23); 3.运用反 ...

  2. 和我一起学Effective Java之创建和销毁对象

    前言 主要学习创建和销毁对象: 1.何时以及如何创建对象 2.何时以及如何避免创建对象 3.如何确保它们能够适时地销毁 4.如何管理对象销毁之前必须进行的清理动作 正文 一.用静态工厂方法代替构造器 ...

  3. 《Effective Java》—— 创建与销毁对象

    本篇主要总结的是<Effecticve Java>中关于创建和销毁对象的内容. 比如: 何时以及如何创建对象 何时以及如何避免创建对象 如何确保及时销毁 如何管理对象销毁前的清理动作 考虑 ...

  4. Effective Java - [2. 创建与销毁对象]

    让对象的创建与销毁在掌控中. Item 1: 使用静态工厂方法而非使用构造函数 public static Boolean valueOf(boolean b) { return b ? Boolea ...

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

    小Alan最近看到了<Effective Java>这本书,这本书包含的内容非常丰富,这本书我就不多介绍了,只能默默的说一句,作为一名java开发错过了这本书难免会成为一个小遗憾,所以还是 ...

  6. Java 避免创建不必要的对象

    最好能重用对象而不是在每次需要的时候就创建一个相同功能的新对象.如果对象是不可变的,它就始终可以被重用. String s = new String("stringette"); ...

  7. 【读书笔记】《Effective Java》——创建和销毁对象

    Item 1. 考虑用静态工厂方法替代构造器 获得一个类的实例时我们都会采取一个共有的构造器.Foo x = new Foo(): 同时我们应该掌握另一种方法就是静态工厂方法(static facto ...

  8. [Effective Java]第三章 对所有对象都通用的方法

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  9. Effective Java 第三版——64. 通过对象的接口引用对象

    Tips 书中的源代码地址:https://github.com/jbloch/effective-java-3e-source-code 注意,书中的有些代码里方法是基于Java 9 API中的,所 ...

随机推荐

  1. 命名空间 namespace

    ​ 命名空间是一块程序员可以自己命名的内存区域,用于解决同名冲突的问题. ​ 举例来说,某班及内有三个张三,分别坐在班级的第一排.第三排和最后一排.当老师喊张三时,三个张三都站起来应答,这就是同名冲突 ...

  2. 为Visual Studio 2019设置 更改皮肤

    下载主题插件:Color Themes for Visual Studio 安装插件 下载完成后 关闭vs2019 完成初始化,初始化完成后,再次打开软件进行配置. 卸载插件 点击卸载,然后关闭vs2 ...

  3. kibana操作记录

    GET _search { "query": { "match_all": {} } } GET _cat/nodes GET _cat/health GET ...

  4. Java课程设计---WindowBuilder插件安装

    1 .获取插件地址 WindowBuilder 地址http://www.eclipse.org/windowbuilder/download.php 打开网址后会看到如下 2.在线方式安装插件 根据 ...

  5. AndroidMainifest.xml文件属性

    1 <manifest xmlns:android="http://schemas.android.com/apk/res/android" 2 package=" ...

  6. MySql绿色版的安装

    MySql绿色版的安装 1.下载 2.配置环境变量,执行命令 mysqld --initialize-insecure --user=mysql 会创建出data文件夹 3.创建my.ini文件 [c ...

  7. oracle 中的decode函数

    decode(条件,值1,返回值1,值2,返回值2,...值n,返回值n,缺省值) 该函数的含义如下:IF 条件=值1 THEN RETURN(翻译值1)ELSIF 条件=值2 THEN RETURN ...

  8. Lua中如何实现类似gdb的断点调试--03通用变量修改及调用栈回溯

    在前面两篇01最小实现及02通用变量打印中,我们已经实现了设置断点.删除断点及通用变量打印接口. 本篇将继续新增两个辅助的调试接口:调用栈回溯打印接口.通用变量设置接口.前者打印调用栈的回溯信息,后者 ...

  9. IDEA在同一工作空间导入多个项目

    在IDEA中导入项目的基本操作是: File--Open--打开工作空间选择要导入的项目 一个项目导入到工作空间啦 但是当你重复这个操作想在这个工作空间继续导入其他模块的时候,就会出现以下问题 不不管 ...

  10. 被vector动态扩容给坑了!

    大家好,我是东北码农.记录一下工作中事. 前几天,运维同事给我反馈了一个问题: 通过监控发现,线上的一个服务,业务线程时不时会出现卡顿,卡顿大约持续几秒. 我们做金融系统后台开发的,对性能要求很严格的 ...