第四条:通过私有构造器强化不可实例化的能力

有时可能需要编写只包含静态方法和静态域的类,这样的工具类不希望被实例化,因为实例化对它来说没有意义。

然而,在缺少显式构造器的情况下,系统会自动提供一个缺省构造,但这种类又不能设计为抽象类,因为它不希望被继承,也不希望它的子类能实例化。

所以,可以为其提供一个显式的私有构造器,保证不能从类的外部调用构造,代码如下:

public class UtilityClass{
private UtilityClass(){
throw new AssertionError(); //AssertionError可以避免不小心在类内调用构造
} ...... }

这种习惯做法也有副作用,它使得一个类不能被子类化。

第五条:优先考虑依赖注入来引用资源

第六条:避免创建不必要的对象

当你应该重用现有对象的时候,请不要创建新的对象。

一般来说,最好能重用单个对象,而不是在每次需要的时候就创建一个新对象,如果对象是不可变的(详见17条),它就始终可以被重用。

反面例子如下:

String s = new String("wuhu");

该语句每次执行的时候都创建一个新的String实例,但这是不必要的,若该语句处于循环中,就会创造出成千上万不必要的String实例。

所以,正确的应该这样做:

String s = "wuhu";

这个版本只用了一个String实例,而且可以保证,对于所有在同一台虚拟机中运行的代码,只要字符串相同,该对象就会被重用。

对于同时提供了静态工厂方法和构造器的不可变类,通常优先使用静态工厂方法而不是构造器,以避免创建不必要的对象。例如,静态工厂方法Boolean valueOf(String)几乎总是优先于构造器Boolean(String)(该构造在java9中已被废弃)。构造器每次调用都会创建一个对象,而静态工厂方法不会。除了重用不可变的对象之外,也可以重用那些已知不会被修改的可变对象。

静态工厂方法详见此处关于 Java 的静态工厂方法,看这一篇就够了! - 简书 (jianshu.com)

有些对象创建成本很高,如果需要重复使用,建议将其缓存下来重用。如以下实例:

static boolean isRomanNumeral(String s){
return s,matches(....); //此处为正则表达式
}

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

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

public class RomanNumerals{
private static final Pattern ROMAN = Pattern.compile(...);
static boolean isRomanNumeral(String s){
return ROMAN.matcher(s).matches();
}
}

如果一个对象是不变的,那么它显然能够被安全的重用,但其他有些情形则并不这么明显。考虑适配器的情形,有时也叫做视图。

适配器模式详见适配器模式 | 菜鸟教程 (runoob.com)

例如,Map接口的keyset方法返回Map对象的Set视图,其中包含该Map中所有的键,对于给定的Map对象,每次调用keySet实际上都返回同样的Set实例,虽然返回的Set实例一般是可改变的,但是所有返回的对象在功能上都是等同的:其中一个返回对象发生变化时,所有其他的返回对象也要发生变化,因为它们是由同一个Map实例支撑的。

另一种创建多余对象的方法,称为自动装箱。请看如下代码:

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

这段程序算出的答案是正确的,但是比实际情况要更慢一点,因为变量sum被声明为Long而不是long,意味着每次先Long sum中增加一个long时都要构造一个实例,所以:要优先使用基本类型而不是装箱基本类型,要当心无意识的自动装箱。

Effective java 读书笔记(2)的更多相关文章

  1. Effective Java读书笔记完结啦

    Effective Java是一本经典的书, 很实用的Java进阶读物, 提供了各个方面的best practices. 最近终于做完了Effective Java的读书笔记, 发布出来与大家共享. ...

  2. Effective java读书笔记

    2015年进步很小,看的书也不是很多,感觉自己都要废了,2016是沉淀的一年,在这一年中要不断学习.看书,努力提升自己 计在16年要看12本书,主要涉及java基础.Spring研究.java并发.J ...

  3. Effective Java 读书笔记(一):使用静态工厂方法代替构造器

    这是Effective Java第2章提出的第一条建议: 考虑用静态工厂方法代替构造器 此处的静态工厂方法并不是设计模式,主要指static修饰的静态方法,关于static的说明可以参考之前的博文&l ...

  4. Effective Java 读书笔记之九 并发

    一.访问共享的可变数据时要同步 1.synchronized关键字既然保证访问的可见性也能保证原子性.而volatile修饰符只能保证变量的线程可见性. 2.增量操作符等不是原子性,多线程操作时可能导 ...

  5. Effective Java 读书笔记之七 通用程序设计

    一.将局部变量的作用域最小化 1.在第一次使用变量的地方声明 2.几乎每个变量的声明都应该包含一个初始化表达式:try-catch语句是一个例外 3.使方法小而集中是一个好的策略 二.for-each ...

  6. Effective Java 读书笔记之一 创建和销毁对象

    一.考虑用静态工厂方法代替构造器 这里的静态工厂方法是指类中使用public static 修饰的方法,和设计模式的工厂方法模式没有任何关系.相对于使用共有的构造器来创建对象,静态工厂方法有几大优势: ...

  7. [Effective Java读书笔记] 第二章 创建和销毁对象(1~7)

    我的技术博客经常被流氓网站恶意爬取转载.请移步原文:http://www.cnblogs.com/hamhog/p/3537576.html,享受整齐的排版.有效的链接.正确的代码缩进.更好的阅读体验 ...

  8. Effective Java读书笔记——第三章 对于全部对象都通用的方法

    第8条:覆盖equals时请遵守通用的约定 设计Object类的目的就是用来覆盖的,它全部的非final方法都是用来被覆盖的(equals.hashcode.clone.finalize)都有通用约定 ...

  9. Effective Java 读书笔记(五):Lambda和Stream

    1 Lamdba优于匿名内部类 (1)DEMO1 匿名内部类:过时 Collections.sort(words, new Comparator<String>() { public in ...

  10. Effective Java 读书笔记(四):泛型

    1 不要使用原始类型 (1)术语 术语 例子 参数化类型(Parameterized type) List<String> 实际类型参数(Actual type parameter) St ...

随机推荐

  1. 开源ASR服务器vosk

    概述 近几年由于AI的迅速发展,语音相关的自然语言处理NLP项目也变多了,新的技术也越来越成熟,其中TTS(语音生成)和ASR(语音识别)是NLP中非常重要的环节. 今天我们介绍一个开源的ASR项目v ...

  2. Docker系列(22)- DockerFile指令说明并构建自己的centos

    DockerFile常用指令 实战测试 DockerHub中99%镜像都是从这个基础镜像过来的FROM scratch,然后配置需要的软件和配置来进行构建 #创建一个自己的centos # 1.编写D ...

  3. Shell系列(29)- 单分支if语句格式

    单分支if条件语句 if [ 条件判断 ] ;then 程序 fi 或者 if [ 条件判断 ] then 程序 fi 注意点 if语句使用fi结尾,和一般语言使用大括号结尾不同 [ 条件判断 ]就是 ...

  4. 『GoLang』语法基础

    标识符 字母或下划线开头 之后只能出现数字.字母.下划线 大小写敏感 Go语言关键字 break default func interface select case defer go map str ...

  5. 在反序列化数据的时候报错raise JSONDecodeError("Expecting value", s, err.value) from None json.decode

    今天在爬取某网站数据内容适合,通过正则匹配拿到了需要的内容字符串,但是在反序列化的时候竟然报错,大概意思知道他不是json的期望值,那么我就会像是不是数据内有一些内容是由于编码的问题导致的呢?因为之前 ...

  6. 鸿蒙内核源码分析(进程回收篇) | 老父亲如何向老祖宗临终托孤 ? | 百篇博客分析OpenHarmony源码 | v47.01

    百篇博客系列篇.本篇为: v47.xx 鸿蒙内核源码分析(进程回收篇) | 临终前如何向老祖宗托孤 | 51.c.h .o 进程管理相关篇为: v02.xx 鸿蒙内核源码分析(进程管理篇) | 谁在管 ...

  7. CF1119H-Triple【FWT】

    正题 题目链接:https://www.luogu.com.cn/problem/CF1119H 题目大意 \(n\)个可重集,第\(i\)个里有\(x\)个\(a_i\),\(y\)个\(b_i\) ...

  8. Python3入门系列之-----算术运算符|比较运算符|赋值运算符|逻辑运算符|成员运算符|身份运算符

    什么是运算符? 本章节主要说明Python的运算符.举个简单的例子 1 +2 = 3 . 例子中,1 和 1.2 被称为操作数,"+" 称为运算符. Python语言支持以下类型的 ...

  9. VulnHub 实战靶场Breach-1.0

    相比于CTF题目,Vulnhub的靶场更贴近于实际一些,而且更加综合考察了知识.在这里记录以下打这个靶场的过程和心得. 测试环境 Kali linux IP:192.168.110.128 Breac ...

  10. C++核心编程 3 函数提高

    函数默认参数 在C++中,函数的形参列表中的形参是可以有默认值的: 语法:返回值类型 函数名 (参数 = 默认值){ } 实例: #include<iostream> using name ...