Effective Java笔记(6)避免创建不必要的对象

news2024/12/23 11:50:27

        一般来说,最好能重用单个对象,而不是在每次需要 的时候就创建一个相同功能的新对象 。 重用方式既快速,又流行 。 如果对象是不可变的( immutable ) (详见第 17 条),它就始终可以被重用 。

        作为一个极端的反面例子,看看下面的语句 :

String s = new String("bikini"); // DON'T DO THIS!

        该语句每次被执行的时候都创建一个新的 String 实例,但是这些创建对象的动作全都是不必要的。 传递给 String构造器的参数(bikini)本身就是一个 String 实例,功能方面等同于构造器创建的所有对象 。 如果这种用法是在一个循环中,或者是在一个被频繁调用的方法中,就会创建出成千上万不必要的 String 实例 。

        改进后的版本如下所示 :

String s = "bikini";

        这个版本只用了 一个 String 实例,而不是每次执行的 时候都创建一个新的实例 。 而且,它可以保证,对于所有在同一台虚拟机中运行的代码,只要它们包含相同的字符串字面常量,该对象就会被重用。

        对于同时提供了静态 工厂方法( static factory method) 和构造器的不可变类,通常优先使用静态工厂方法而不是构造器,以避免创建不必要的对象 。 例如,静态工厂方法Boolean.valueOf(String )几乎总是优先于构造器 Boolean(String),注意构造器 Boolean(String)在 Java 9 中已经被废弃了 。 构造器在每次被调用的时候都会创建一个新的对象,而静态工厂方法则从来不要求这样做,实际上也不会这样做 。 除了重用不可变的对象之外,也可以重用那些已知不会被修改的可变对象。

        有些对象创建的成本比其他对象要高得多 。 如果重复地需要这类“昂贵的对象”,建议将它缓存下来重用 。 遗憾的是,在创建这种对象的时候,并非总是那么显而易见 。 假设想要编写一个方法,用它确定一个字符串是否为一个有效的罗马数字 。 下面介绍一种最容易的方法,使用一个正则表达式 :

        这个实现的问题在于它依赖 String.matches 方法 。虽然String.matches 方法最易于查看一个字符串是否与正则表达式相匹配 , 但并不适合在注重性能的情形中重复使用 。 问题在于,它在内部为正则表达式创建了一个 Pattern 实例,却只用了一次,之后就可以进行垃圾回收了 。 创建Pattern实例的成本很高 ,因为需要将正则表达式编译成一个有限状态机( finite state machine)。

        为了提升性能,应该显式地将正则表达式编译成一个 Pattern实例(不可变),让它成为类初始化的一部分,并将它缓存起来,每当调用 isRomanNumeral 方法的时候就重用同一个实例:

         改进后的 isRomanNumeral 方法如果被频繁地调用,会显示出明显的性能优势。

        如果一个对象是不变的,那么它显然能够被安全地重用,但其他有些情形则并不总是这么明显 。 考虑适配器( adapter )的情形,有时也叫作视图( view ) 。 适配器是指这样一个对象 :它把功能委托给一个后备对象( backing object ),从而为后备对象提供一个可以替代的接口 。 由于适配器除了后备对象之外, 没有其他的状态信息,所以针对某个给定对象的特定适配器而言,它不需要创建多个适配器实例 。

        例如 ,Map 接口 的 keySet 方法返回该 Map 对象的 Set 视图,其 中包含该 Map 中所有的键(key) 。 乍看之下 ,好像每次调用 keySet 都应该创建一个新 的 Set 实例,但是,对于一个给定的 Map 对象,实际上每次调用 keySet 都返回同样的 Set 实例 。 虽然被返回的 Set 实例一般是可改变的,但是所有返回的对象在功能上是等同的 :当其中一个返回对象发生变化的时候,所有其他的返回对象也要发生变化,因为它们是由 同一个 Map 实例支撑的 。 虽然创建 key Set 视图对象的多个实例并无害处, 却是没有必要,也没有好处的 。

        另一种创建多余对象的方法,称作自动装箱( autoboxing ),它允许程序员将基本类型和装箱基本类型( Boxed Primitive Type )混用,按需要自动装箱和拆箱 。自动装箱使得基本类型和装箱基本类型之间的差别变得模糊起来, 但是并没有完全消除 。 它们在语义上还有着微妙的差别,在性能上也有着比较明显的差别 。 请看下面的程序,它计算所有int正整数值的总和 。 为此,程序必须使用 long 算法,因为 int 不够大,无法容纳所有 int 正整数值的总和:

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

        这段程序算出的答案是正确的,但是比实际情况要更慢一些,只因为打错了一个字符 。 变量 sum 被声明成 Long 而不是 long ,意味着程序构造了大约 231 个多余的 Long 实例(大约每次往 Long sum 中增加 long时构造一个实例) 。 将 sum 的声明从 Long 改成long ,在我的机器上使运行时间从 6.3 秒减少到了 0 . 59 秒 。 结论很明显 :要优先使用 基本类型而不是装箱基本类型,要当心无意识的自动装箱

        不要错误地认为本条目所介绍的内容暗示着“创建对象的代价非常昂贵,我们应该要尽可能地避免创建对象” 。 相反,由于小对象的构造器只做很少量的显式工作,所以小对象的创建和回收动作是非常廉价的,特别是在现代的 JVM 实现上更是如此 。 通过创建附加的对象,提升程序的清晰性、简洁性和功能性,这通常是件好事 。

        反之,通过维护自己的对象池( object pool )来避免创建对象并不是一种好的做法,除非池中的对象是非常重量级的 。 正确使用对象池的典型对象示例就是数据库连接池 。 建立数据库连接的代价是非常昂贵的,因此重用这些对象非常有意义 。 而且,数据库的许可可能限制你只能使用一定数量的连接 。 但是,一般而言,维护自己的对象池必定会把代码弄得很乱,同时增加内存占用( footprint ),并且还会损害性能 。 现代的 JVM 实现具有高度优化的垃圾回收器,其性能很容易就会超过轻量级对象池的性能 。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/748460.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Golang gui walk入门教程(一)安装walk环境

一、golang环境 Go 1.11.x or later 二、安装walk go get github.com/lxn/walk 三、安装rsrc 运行walk程序需要manifest&#xff0c;rsrc提供了这个功能 go install github.com/akavel/rsrc 安装完成后在GOPATH的bin下面会有一个rsrc.exe的可执行文件 在idea的termial输入r…

Tauri 提供界面 + 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令

Tauri 提供界面 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令 文章目录 Tauri 提供界面 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令一、Tauri 概述二、界面预览三、代码参考1、main.rs2、App.vue3、Greet.vue4、依赖 一、Tauri 概述 Tauri 是一…

C语言动态获取设备的网络接口名称和状态以及对应的IP地址

一、目的 在实际项目中需要获取设备的IP地址然后通过广播的形式通知局域网内的其他设备。 二、介绍 方法一 通过ioctl方式获取SIOCGIFADDR信息 /** C Program to Get IP Address*/ #include <stdio.h> #include <string.h> #include <sys/types.h> #includ…

mfc120u.dll丢失修复,mfc120u.dll缺失的解决方法

MFC120u.dll缺失的原因 当系统中缺少或损坏了MFC120u.dll文件时&#xff0c;就会出现"MFC120u.dll缺失"的错误提示。造成MFC120u.dll缺失的原因可能有以下几种情况&#xff1a; 1.文件删除或损坏&#xff1a;MFC120u.dll文件可能因为误删除、病毒感染、硬盘故障等原…

pearsonr报错:计算结果为nan、warning .warn (stats.constantinputwarning (msg))

【1】两个都是ndarry 最后结果为nan&#xff1a; &#xff08;1&#xff09;数据类型转换&#xff1a;都转为一样的float32&#xff1a;依旧报错nan &#xff08;2&#xff09;进入函数内部debug&#xff1a; if (xx[0]).all() or (yy[0]).all() warning .warn (stats.consta…

13-接口、代码、命令如何测试?

可以通过工具&#xff0c;也可以通过代码来进行测试。 这里使用工具Postman。 1.针对接口进行测试 粘贴导入到Postman中&#xff1a; 也可直接复制URL&#xff1a; 粘贴到Postman中&#xff1a; 没有对应的参数&#xff0c;那么可以直接请求&#xff1a; 这个接口的GET、POST…

无法将“pip“识别为cmdlet、函数、脚本文件或可运行程序的名称。

出现问题如下&#xff1a; 出现问题原因&#xff1a; 没有添加pip对应的安装目录进入环境变量里面的系统变量。 解决方案&#xff1a; 1.确定python的安装路径 将python的路径添加到系统变量中 2.输入pip所在的安装路径&#xff1a; python路径\Lib\site-packages 3.添加…

PyTorch训练RNN, GRU, LSTM:手写数字识别

文章目录 pytorch 神经网络训练demoResult参考来源 pytorch 神经网络训练demo 数据集&#xff1a;MNIST 该数据集的内容是手写数字识别&#xff0c;其分为两部分&#xff0c;分别含有60000张训练图片和10000张测试图片 图片来源&#xff1a;https://tensornews.cn/mnist_intr…

eureka使用错误

错误 java.net.UnknownHostException: INVENTORYSERVICE 分析&解决&#xff1a; 这里的请求执行错误 但eureka可以找到服务 手动创建RestTemlate到容器中&#xff0c;未加LoadBalanced注解 加上注解后重试&#xff0c;成功

java详细显示try/catch块的异常类方法等信息

示例&#xff1a; Testpublic void testException(){try {double theorynumDouble Double.parseDouble(null);} catch (Exception e) {StackTraceElement[] stackTraceElements e.getStackTrace();StackTraceElement stackTraceElementFirst stackTraceElements[0];String c…

一文教你学会Linux数组

目录 &#x1f380;什么是数组&#xff1f; &#x1f380;数组优点 &#x1f380;数组缺点 &#x1f380;定义数组 &#x1f380;数组的取值 &#x1f380;一次取出数组所有的值 &#x1f380;数组长度&#xff1a; 即数组元素个数 &#x1f380;数组的截取&#xff…

【ASP.NET】医学实验室管理(LIS)系统源码

一、医学实验室LIS系统概况 LIS&#xff08;全称Laboratory Information Management System&#xff09;&#xff0c;是专为医院检验科设计的一套实验室信息管理系统&#xff0c;能将实验仪器与计算机组成网络&#xff0c;使病人样品登录、实验数据存取、报告审核、打印分发&am…

MySQL数据库的索引原理与慢SQL优化的5大原则

这篇文章主要介绍了MySQL数据库的索引原理与慢SQL优化的5大原则,包括&#xff1a;建立索引的原则&#xff0c;慢查询优化基本步骤&#xff0c;慢查询优化案例&#xff0c;explain使用&#xff0c;需要的朋友可以参考下 我们知道一般的应用系统&#xff0c;读写比例在10:1左右&…

计算机系统结构与操作系统实验三(2)-进入保护模式

&#x1f4cd;实验要求 从实模式到保护模式的转变&#xff1a; 在刚进入loader尚在实模式下时&#xff0c;在第2行显示&#xff1a;real-Zhangsan 在变为保护模式后&#xff0c;在第3行显示&#xff1a;protect-Zhangsan 加载gdt、将cr0的PE位置为1 &#x1f4cd;实验过程 老…

建议收藏 | 可视化ETL平台--Kettle

Kettle的介绍 学习目标 知道什么是ETL及Kettel是开源的ETL工具 了解kettle环境的安装流程 1.ETL介绍 ETL&#xff08;Extract-Transform-Load的缩写&#xff0c;即数据抽取、转换、装载的过程&#xff09;&#xff0c;对于企业或行业应用来说&#xff0c;我们经常会遇到各种…

redis可视化工具 Another Redis Desktop Manager

Redis API 官方文档&#xff1a;Redis 命令参考 — Redis 命令参考 Redis 可视化工具下载&#xff1a; Releases qishibo/AnotherRedisDesktopManager GitHub&#xff08;github&#xff09; 使用&#xff1a; 1、新建连接 点击 新建连接 创建新的连接 切换语言 地址&a…

前端Vue自定义商品订单星级评分 爱心评分组件

随着技术的发展&#xff0c;开发的复杂度也越来越高&#xff0c;传统开发方式将一个系统做成了整块应用&#xff0c;经常出现的情况就是一个小小的改动或者一个小功能的增加可能会引起整体逻辑的修改&#xff0c;造成牵一发而动全身。 通过组件化开发&#xff0c;可以有效实现…

数据分类分级的概念、方法、标准及行业实践

数据战略上升为国家战略&#xff0c;数据资产成为国家各行各业的核心资产。在数字化时代&#xff0c;数据分类分级成为数据资产管理的重要组成部分。 通过数据分类分级管理&#xff0c;可有效使用和保护数据&#xff0c;使数据更易于定位和检索&#xff0c;满足数据风险管理、…

领略大数据分析的魅力:迈向FineBI的世界

版权声明&#xff1a;本文为 小异常 原创文章&#xff0c;非商用自由转载-保持署名-注明出处&#xff0c;谢谢&#xff01; 本文网址&#xff1a;https://sunkuan.blog.csdn.net/article/details/131698171 文章目录 一、FineBI介绍&#xff08;6.0版本&#xff09;1、使用人群…

高并发的哲学原理(三)-- 基础设施并发:虚拟机与 Kubernetes(k8s)

上篇文章说到&#xff0c;Apache 无法处理海量用户的 TCP 连接&#xff0c;那要是由于宇宙时空所限&#xff0c;你的系统就是无法离开 Apache&#xff0c;该怎么承接高并发呢&#xff1f;有办法&#xff1a;既然单机不行&#xff0c;那就把单机虚拟化成多个 Linux 机器&#xf…