Effective java 读书笔记(2)
第四条:通过私有构造器强化不可实例化的能力
有时可能需要编写只包含静态方法和静态域的类,这样的工具类不希望被实例化,因为实例化对它来说没有意义。
然而,在缺少显式构造器的情况下,系统会自动提供一个缺省构造,但这种类又不能设计为抽象类,因为它不希望被继承,也不希望它的子类能实例化。
所以,可以为其提供一个显式的私有构造器,保证不能从类的外部调用构造,代码如下:
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)的更多相关文章
- Effective Java读书笔记完结啦
Effective Java是一本经典的书, 很实用的Java进阶读物, 提供了各个方面的best practices. 最近终于做完了Effective Java的读书笔记, 发布出来与大家共享. ...
- Effective java读书笔记
2015年进步很小,看的书也不是很多,感觉自己都要废了,2016是沉淀的一年,在这一年中要不断学习.看书,努力提升自己 计在16年要看12本书,主要涉及java基础.Spring研究.java并发.J ...
- Effective Java 读书笔记(一):使用静态工厂方法代替构造器
这是Effective Java第2章提出的第一条建议: 考虑用静态工厂方法代替构造器 此处的静态工厂方法并不是设计模式,主要指static修饰的静态方法,关于static的说明可以参考之前的博文&l ...
- Effective Java 读书笔记之九 并发
一.访问共享的可变数据时要同步 1.synchronized关键字既然保证访问的可见性也能保证原子性.而volatile修饰符只能保证变量的线程可见性. 2.增量操作符等不是原子性,多线程操作时可能导 ...
- Effective Java 读书笔记之七 通用程序设计
一.将局部变量的作用域最小化 1.在第一次使用变量的地方声明 2.几乎每个变量的声明都应该包含一个初始化表达式:try-catch语句是一个例外 3.使方法小而集中是一个好的策略 二.for-each ...
- Effective Java 读书笔记之一 创建和销毁对象
一.考虑用静态工厂方法代替构造器 这里的静态工厂方法是指类中使用public static 修饰的方法,和设计模式的工厂方法模式没有任何关系.相对于使用共有的构造器来创建对象,静态工厂方法有几大优势: ...
- [Effective Java读书笔记] 第二章 创建和销毁对象(1~7)
我的技术博客经常被流氓网站恶意爬取转载.请移步原文:http://www.cnblogs.com/hamhog/p/3537576.html,享受整齐的排版.有效的链接.正确的代码缩进.更好的阅读体验 ...
- Effective Java读书笔记——第三章 对于全部对象都通用的方法
第8条:覆盖equals时请遵守通用的约定 设计Object类的目的就是用来覆盖的,它全部的非final方法都是用来被覆盖的(equals.hashcode.clone.finalize)都有通用约定 ...
- Effective Java 读书笔记(五):Lambda和Stream
1 Lamdba优于匿名内部类 (1)DEMO1 匿名内部类:过时 Collections.sort(words, new Comparator<String>() { public in ...
- Effective Java 读书笔记(四):泛型
1 不要使用原始类型 (1)术语 术语 例子 参数化类型(Parameterized type) List<String> 实际类型参数(Actual type parameter) St ...
随机推荐
- 前端常用场景总结CSS/JS/插件(实用篇更新中...)
<div class="box box1"> <span>垂直居中</span> </div> .box1{ display: ta ...
- Shell系列(24)- 条件判断之文件类型
按照文件类型进行判断 标红,记住:其他了解即可 测试选项 作用 -b 文件 判断该文件是否存在,并且是否为块设备文件(是块设备文件为真) -c 文件 判断该文件是否存在,并且是否为字符设备文件(是字符 ...
- 搞不定 NodeJS 内存泄漏?先从了解垃圾回收开始
通常来说,内存管理有两种方式,一种是手动管理,一种是自动管理. 手动管理需要开发者自己管理内存,什么时候申请内存空间,什么时候释放都需要小心处理,否则容易形成内存泄漏和指针乱飞的局面.C 语言开发是典 ...
- 简单易行的美化方案:Ubuntu 18.04 把启动过程中的紫色美化为黑色
背景 给笔记本装了一个Ubuntu,嫌弃启动的颜色很丑:因此在网上找到了一些修改方法,集成为一个傻瓜脚本. 参考文档: https://askubuntu.com/questions/5065/how ...
- IDEA快捷键(未使用)
1.ctrl Ctrl + Y 删除光标所在行 或 删除选中的行 Ctrl + W 递进式选择代码块.可选中光标所在的单词或段落,连续按会在原有选中的基础上再扩展选中范围 Ctrl + E 显示最近打 ...
- Linux命令行:free
total used free shared buff/cache availableMem: 251G ...
- NOIP 模拟四 考试总结
#T1随 又是liu_................... 数列n,m个操作,每次随机取a[i],x=x*a[i]%k; 问题是求x期望%mod; 首先考虑到期望转移过程中存在%k,一般套路线性期望 ...
- The art of multipropcessor programming 读书笔记-硬件基础2
本系列是 The art of multipropcessor programming 的读书笔记,在原版图书的基础上,结合 OpenJDK 11 以上的版本的代码进行理解和实现.并根据个人的查资料以 ...
- Oracle数据泵数据迁移
1 表空间查询 1.1 检查用户与表空间对应情况 select username,default_tablespace from dba_users; 1.2 查看临时表空间 select ta ...
- 用C++实现的数独解题程序 SudokuSolver 2.3 及实例分析
SudokuSolver 2.3 程序实现 用C++实现的数独解题程序 SudokuSolver 2.2 及实例分析 里新发现了一处可以改进 grp 算法的地方,本次版本实现了对应的改进 grp 算法 ...