【第4条】避免创建不必要的对象

2024-04-17 22:48
文章标签 创建 对象 避免 不必要

本文主要是介绍【第4条】避免创建不必要的对象,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

此条在中文版第二版中被译为了“避免创建不必要的对象”,用此更加严谨了。

 

此条认为重复使用同一对象,比每次需要时都创建一个功能上相等价的新对象更好。如果对象是非可变的(见【第13条】),那么他总是可以被重用的。

 

一例子是:

String s1 = "Hello World !";
........  // 其他一些列代码
String s2= "Hello World !";

 虽然看起来s2是新创建的一个新对象,但是由于String是非可变的,所以 String s2= "Hello World !";
 等同于 String s2 = s1; 实际内存中并没有新开辟一块空间,而是将s2的地址指针指向s1的空间地址。

 

但是如果写成  String s2 = new String(s1); 则系统会为s2新开辟一块内存空间,这是我们所不希望看到的。

进而举个更为极端的反例: String s = new String("Hello !");

由于实参"Hello !"本身就是一个String的实例,这时候实际上已经创建了一个String的实例了,再用之作为new String()的参数,就再次创建了一个String实例给s,于是内存的消耗就Double了!

 

    对于提供了静态工厂方法(见【第1条】)和构造函数的非可变类,你通常可以利用静态工厂方法而不是构造函数,以避免重复创建对象。例如,Boolean.valueOf(String)几乎总是优先于构造函数Boolean(String)的。因为构造函数每次被调用都会创建一个新的对象,而静态工厂方法从来不要求这样做。

 

    再有就是那些已知不会被修改的对象,它们一旦被计算出来就不再变化,可以随时使用,而不必每次使用前再次计算。例如,有一个SystemInfo类,里面有保存和取得系统信息的域和方法。为了简单我们认为只有一个用于获取和保持CPUID的方法和域。如果这个类写的不好的话,可能在每次调用 getCPUID()时,都要重新创建SystemInfo中的cpuId,重新获取,甚至重新创建一个SystemInfo类;而好的代码,是在第一次调用时获取,并保持起来,之后再次调用时只是返回就可以了。

以下是我的一段真实代码中的截取:

public class SystemInfo {private static final SystemInfo INSTANCE = new SystemInfo();private static List<String> NULL_STRING_LIST;   // 应该是一个final常量,但由于这里的执行顺序在构造函数之后,所以要在被构造函数调用的方法中使用这个常量就不可以了,所以才这样写// 是用List是因为CPU可能不止一个private List<String> cpuIds;/*** 这是一个私有的构造函数 目的是禁止此类被外部实例化,因为这是一个单例模式*/private SystemInfo() {NULL_STRING_LIST = new ArrayList<String>();this.cpuIds = getCPUIDs();}public static SystemInfo getInstance() {return INSTANCE;}public List<String> getCPUID() {return this.cpuIds;}private List<String> getCPUIDs() {try {return Arrays.asList(...);} catch (Exception e) {return NULL_STRING_LIST; // 见【第27条】返回零长度的数组而不是null,推想一下ArrayList也是同理}}
}

   

使用方法: String myCpu = SystemInfo.getInstance().getCPUID();

这个例子综合了【前4条】的知识,并且还包括了【第27条】,是我能想出的比较好的例子了。

【第1条】它提供了静态工厂方法,用于代替公有构造函数

【第2条】它使用私有构造函数来强化它是个单例模式

【第3条】它使用私有构造函数来强化不可实例化的能力(但与第3条不完全相同,它允许内部实例化一次)

【第27条】异常时返回零长度的数组而不是null,避免使用者需要过多的保护性代码

 

 

 进而这段代码的一个细微变形版本如下,不再是单例模式,而是不可实例化的工具类了。(是【第2条】的进一步,完全满足【第3条】)

public class SystemInfo {private static final SystemInfo INSTANCE = new SystemInfo();private static final List<String> NULL_STRING_LIST = new ArrayList<String>();// 是用List是因为CPU可能不止一个private static List<String> cpuIds;/*** 这是一个空的私有的构造函数 目的是禁止此类被实例化,因为这是一个工具类  */private SystemInfo() {}static{cpuIds = getCPUIDs();}public static SystemInfo getInstance() {return INSTANCE;}public List<String> getCPUID() {return cpuIds;}private static List<String> getCPUIDs() {try {return Arrays.asList(...);} catch (Exception e) {return NULL_STRING_LIST; // 见【第27条】返回零长度的数组而不是null,推想一下ArrayList也是同理}}
}

 

以上两段哪种写法更好呢?可能仁者见仁吧,实际中我使用的是后者,而且是把这个类实现了一个接口,因为考虑到可能会有不同版本的“取得系统信息”的方法(Win32的、Linux的、JNI的...)。

而且当使用Spring的注入后,getInstance()方法 和 INSTANCE 常量也不再是必须的了(Spring的注入不在现在的讨论之内)

 

 

 

【Effective Java 学习笔记】系列连载专题请见:
http://tonylian.iteye.com/categories/64208

 

这篇关于【第4条】避免创建不必要的对象的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/913075

相关文章

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定

Java 创建图形用户界面(GUI)入门指南(Swing库 JFrame 类)概述

概述 基本概念 Java Swing 的架构 Java Swing 是一个为 Java 设计的 GUI 工具包,是 JAVA 基础类的一部分,基于 Java AWT 构建,提供了一系列轻量级、可定制的图形用户界面(GUI)组件。 与 AWT 相比,Swing 提供了许多比 AWT 更好的屏幕显示元素,更加灵活和可定制,具有更好的跨平台性能。 组件和容器 Java Swing 提供了许多

顺序表之创建,判满,插入,输出

文章目录 🍊自我介绍🍊创建一个空的顺序表,为结构体在堆区分配空间🍊插入数据🍊输出数据🍊判断顺序表是否满了,满了返回值1,否则返回0🍊main函数 你的点赞评论就是对博主最大的鼓励 当然喜欢的小伙伴可以:点赞+关注+评论+收藏(一键四连)哦~ 🍊自我介绍   Hello,大家好,我是小珑也要变强(也是小珑),我是易编程·终身成长社群的一名“创始团队·嘉宾”

Maven创建项目中的groupId, artifactId, 和 version的意思

文章目录 groupIdartifactIdversionname groupId 定义:groupId 是 Maven 项目坐标的第一个部分,它通常表示项目的组织或公司的域名反转写法。例如,如果你为公司 example.com 开发软件,groupId 可能是 com.example。作用:groupId 被用来组织和分组相关的 Maven artifacts,这样可以避免

Java第二阶段---09类和对象---第三节 构造方法

第三节 构造方法 1.概念 构造方法是一种特殊的方法,主要用于创建对象以及完成对象的属性初始化操作。构造方法不能被对象调用。 2.语法 //[]中内容可有可无 访问修饰符 类名([参数列表]){ } 3.示例 public class Car {     //车特征(属性)     public String name;//车名   可以直接拿来用 说明它有初始值     pu

批处理以当前时间为文件名创建文件

批处理以当前时间为文件名创建文件 批处理创建空文件 有时候,需要创建以当前时间命名的文件,手动输入当然可以,但是有更省心的方法吗? 假设我是 windows 操作系统,打开命令行。 输入以下命令试试: echo %date:~0,4%_%date:~5,2%_%date:~8,2%_%time:~0,2%_%time:~3,2%_%time:~6,2% 输出类似: 2019_06

ORACLE 11g 创建数据库时 Enterprise Manager配置失败的解决办法 无法打开OEM的解决办法

在win7 64位系统下安装oracle11g,在使用Database configuration Assistant创建数据库时,在创建到85%的时候报错,错误如下: 解决办法: 在listener.ora中增加对BlueAeri-PC或ip地址的侦听,具体步骤如下: 1.启动Net Manager,在“监听程序”--Listener下添加一个地址,主机名写计

如何来避免FOUC

FOUC(Flash of Unstyled Content)是指在网页加载过程中,由于CSS样式加载延迟或加载顺序不当,导致页面出现短暂的无样式内容闪烁现象。为了避免FOUC,可以采取以下几种方法: 1. 优化CSS加载 内联CSS:将关键的CSS样式直接嵌入到HTML文档的<head>部分,这样可以确保在页面渲染之前样式就已经加载和应用。提前引入CSS:将CSS文件放在HTML文档的<he

HTML5自定义属性对象Dataset

原文转自HTML5自定义属性对象Dataset简介 一、html5 自定义属性介绍 之前翻译的“你必须知道的28个HTML5特征、窍门和技术”一文中对于HTML5中自定义合法属性data-已经做过些介绍,就是在HTML5中我们可以使用data-前缀设置我们需要的自定义属性,来进行一些数据的存放,例如我们要在一个文字按钮上存放相对应的id: <a href="javascript:" d