《Effective Java》总结
导语
《Effective Java》是和《Thinking in java》齐名的java进阶书籍。作者参与了JDK标准库的编写工作,对于此书的学习,让我收获很多。好记性不如烂笔头,我决定好好总结一下。本书主要内容有11章,分别从各个方面阐述了作者对于java代码编写的体会。我看的是第二版,目前最新版已经是第三版了,但是还没有在国内翻译出版。这就是英语不好的局限之处~
创建和销毁对象
作者认为,使用构造方法构造对象是不优雅的。我们应该是用工厂方法来构建对象。一来可以用语义化的静态方法名来避免构造方法重载造成的迷惑性,二来可以在构造对象时进行如懒加载,校验等相关控制,因为构造方法调用后一定会返回一个新的分配了堆内存的对象引用。通常的方法名为:
valueOf
,of
,getInstance
,newInstance
,getType
,newType
等。作者认为,对象参数过多时,为了避免冗长的setter方法调用,应该使用Builder设计模式。同时builder对象可以作为参数传递给相关方法。此处可以配合Lombok。
作者认为构造单例的最好方式是枚举,同时应该注意对象序列化对单例造成的破坏。
作者认为只包含静态方法和静态域的类应该私有构造器,因而不可实例化,不能被继承(子类实例化时会调用父类构造器)。个人认为也可以使用abstarct关键字。
作者认为避免创建不必要的对象,如
String s =new String("hello")
,这里其实就重复定义了字符串,造成了内存浪费和GC压力。还有就是循环语句中的某些对象可以在语句外面创建,在整个循环中重用。避免自动装箱造成的频繁创建对象。在我们自己管理内存时,要注意消除过期对象的引用,让JVM及时回收内存。我们自己管理内存的典型场景为我们在全局范围内持有一个数组,我们自己管理自己的数组内存,进行数据的增删改。我们要及时的将不用的对象引用从数组中通过置为
null
去掉。让GC内存回收内存。避免使用finalize方法。在此方法历写业务的最大问题时,该方法的调用时间是不一定的。因为GC线程的优先级很低,所以该方法的调用时机是不可控的,且性能开销很大。只建议在某些特定情况下作为安全网方法,即最后一道捕捉漏网之鱼的方法。
应该重写equals方法。此处可以配合Lombok。
应该同时重写hasCode方法。此处可以配合Lombok。
应该要覆盖toString方法。便于调试。此处可以配合Lombok。
应该谨慎的使用clone方法。因为存在浅克隆问题。
对于值对象类,应该合理的实现
Comparable
方法。此处可以配合guava的Ordering。应该使类的成员可见性在可能的情况下最小。因为一旦类的成员暴露,那就成为创建者和调用者之间的约定。那么类的创建者就要对这些暴露的成员的稳定性负责,不能随意修改约定。
应该通过方法访问域而不是直接访问域。因为可以在方法中进行相关控制。
应该使对象的可变性最小。即,除非必要,所有的域都应该用
final
修饰,所有域都应该是私有的,所有类都不能被拓展。被拓展的类可能会被破坏封装性,即,拓展的过程中,需要去了解被拓展的类的内部实现,而不仅仅是被拓展的类暴露出来的API。复合优于继承。如上所述,继承破坏了封装性,同时复合更稳定,不会受到被复合的类的结构的改变的影响。通过复合和转发(适配器模式)可以方便的实现继承的效果。同时可以通过
asType
方法暴露一个被复合的类的视图模拟多态。继承是一种约定。要么提供丰富和负责任的文档告知方法的自用性,并测试被继承后的风险,要么就通过final禁止被继承。
接口优于抽象类。因为单继承的局限性,同时接口之间并不一定和类一样是分层关系。通过mixin接口(如
Comparable
接口),可以构建功能丰富的类,并且可以灵活的实现多态。为了便捷的实现接口,可以提供一个接口默认实现骨架类(Wapper)。接口只用于定义类型,便于实现多态。不应该用来定义常量,常量属于不同的实现类的内部实现,不应该放在接口中作为API约定。
通过在类中添加一个type字段来区分不同类型是不够OOP的,应该通过类的继承来实现层次。将相同的部分抽象成父类,type字段的每一个值都作为一个拓展子类。添加自己的字段和方法。
用函数对象表示策略,java8之前不支持函数式编程的迂回。
优先考虑静态内部类。静态内部类时访问权限的最小域。如果一个类只在本类中使用,就可以定义为私有的静态内部类。当然,它实现的接口可以有公开的访问权限。
不要使用原生类型(raw type)。对于泛型类,都应该使用参数化类型。
使用
SupressWarn()
压制编译器警告。前提是必须要了解警告的内容,同时将压制的区域最小化,避免出现诡异的问题。列表优于数组。两个原因,数组是协变的,容易产生类型转换异常。数组不支持泛型。
使用泛型。避免类型转换异常。
使用泛型方法。便捷的类型推导可以使代码更简洁。
使用有限制的类型通配符来提高API的灵活性。
Class类是参数化类型,有很多类型相关的方法。如:
cast
等。注意灵活使用。常量推荐使用枚举定义。
不要在业务中使用枚举的
ordinal
属性,应该自定义一个实例域。使用EnumSet来代替位域。
使用EnumMap代替序数索引。枚举作为map的key时。
枚举可以实现接口(因为枚举是一种特殊的类)。
注解优于命名模式。通过命名约定来指定范围,不如在类或类成员上添加注解标记。
使用Override注解来进行编译器检查。
某些只在类上的标记,可以考虑标记接口。如
Cloneable
,Serialble
。检查参数,快速失败。
必要时对参数进行拷贝。如果方法调用者将引用传递给方法后,在方法外对引用指向的对象进行了修改,可能会造成诡异的不可预期的错误。
谨慎设计方法签名。这部分内容太多,也很经典。建议时不时翻下这一节。方法名要合乎约定;不要过分追求方法便捷,除非某个功能太常用可以提供一个便捷方法,不然的话提供一个功能齐全的方法就好;避免过长的参数列表,如有必要,可以提供一个辅助类;注意方法正交性,即功能不要重复。参数优先使用接口,离散参数如布尔参数优先使用枚举。
慎用重载。可以定义一个参数相关的方法名。
慎用可变参数。不安全,代码不美观。建议是一个必要参数加一个可变参数。
返回空数组或空集合,而不是null。
为所有公开API编写文档。
局部变量作用于最小化。
尽可能的使用类库。
精确计算使用大数类型。
基本类型优于包装类型。
尽量避免使用字符串。
小心字符串连接的性能。
接口优于反射机制。反射效率低,无法进行编译期检查。
谨慎的进行优化。前期能跑就行,不要进行没有目的的优化。
不要用异常进行分支控制。
丰富异常信息。
不要忽略异常。
优先使用标准的异常。
并发编程中,尽量使用并发工具类。
序列化没啥用啊,不说了。
总结
我感觉这本书就是很厉害,尤其是带我理解了Super Type Token
这个关于反射和泛型的概念。当然了,那又是一篇博客了,找时间我会娓娓道来的。这本书里关于类和方法的内容很精辟,让我理解了一些关于OOP迷惑。其实值得我多刷好几遍,但是我就匆忙的两三天刷完了。。。心态浮躁。第四章第五章找个午休再翻翻吧。
第一次正式的开始用Markdown写博客,感觉自己很极客。就是博客园的Markdown主题有点太low了。以后我就用这个来写博客啦。。。
后面的博客计划是总结下字节码,反射和泛型。还想着刷一本Zookeeper的书写一写Zookeeper的相关实践。就是心情太浮躁,做事效率太低,拖拖拉拉让我很烦恼。
加油吧。
《Effective Java》总结的更多相关文章
- 《FLASH PROGRAMMING 那些事》总结
注明来自 http://www.ssdfans.com/?p=5589 以MLC为例: 对FGF(Floating Gate Flash)技术的,MLC programming一般分两步走:先prog ...
- Implementation of Serial Wire JTAG flash programming in ARM Cortex M3 Processors
Implementation of Serial Wire JTAG flash programming in ARM Cortex M3 Processors The goal of the pro ...
- [原创] Keil uVision5 下载程序 add flash programming algorithm选项缺少需要的算法解决办法
MDK开发环境从V4升级到V5后,支持包不再是集成到开发环境当中,而是封装在PACK中,需要自行安装,比较麻烦. 搭建MDK开发环境以及破解的方法,在前面的文章中有详细说明,这里不再赘述,有兴趣的可以 ...
- Turtelizer 2 provide JTAG Flash programming and debugging of ARM based boards via USB
http://www.ethernut.de/en/hardware/turtelizer/ Introducing Turtelizer 2 Overview Turtelizer 2 had be ...
- 痞子衡嵌入式:恩智浦i.MX RT1xxx系列MCU硬件那些事(2.3)- 串行NOR Flash下载算法(J-Link工具篇)
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是J-Link工具下i.MXRT的串行NOR Flash下载算法设计. 在i.MXRT硬件那些事系列之<在串行NOR Flash X ...
- Programming Internal Flash Over the Serial Wire Debug <SWD> Interface -- EFM32
1 Debug Interface Overview 1.1 Serial Wire Debug Serial Wire Debug (SWD) is a two-wire protocol for ...
- Error:Flash Download Failed-"Cortex-M3"
Error:Flash Download Failed-"Cortex-M3"出现一般有两种情况: 1.SWD模式下,Debug菜单中,Reset菜单选项(Autodetect/H ...
- STM32F4读写内部FLASH【使用库函数】
STM32F4Discovery开发帮使用的STM32F407VGT6芯片,内部FLASH有1M之多.平时写的代码,烧写完之后还有大量的剩余.有效利用这剩余的FLASH能存储不少数据.因此研究了一下S ...
- Flash Download Failed-"Cortex-M3"
rror:Flash Download Failed-"Cortex-M3"出现一般有两种情况: 1.SWD模式下,Debug菜单中,Reset菜单选项(Autodetect/HW ...
- Stm32_调试出现 Error:Flash Download Failed-"Cortex-M3"
rror:Flash Download Failed-"Cortex-M3"出现一般有两种情况: 1.SWD模式下,Debug菜单中,Reset菜单选项(Autodetect/HW ...
随机推荐
- chrome标签记录——关于各类性能优化
概述 详情 概述 平时经常浏览各大博客,总感觉要学习和需要学习的内容太多太多,而自己的个人能力还不足够写出一些好的文章出来,就只能通过学习他人的东西不断提升自己的实力,然后就会记录收藏各种优秀的博客资 ...
- DDD领域驱动设计:CQRS
1 前置阅读 在阅读本文章之前,你可以先阅读: DDD领域驱动设计是什么 DDD领域驱动设计:实体.值对象.聚合根 DDD领域驱动设计:仓储 MediatR一个优秀的.NET中介者框架 2 什么是CQ ...
- 2019 第十届蓝桥杯大赛软件类省赛 Java A组 题解
2019 第十届蓝桥杯大赛软件类省赛 Java A组 试题A 题解 题目最后一句贴心的提示选手应该使用 long (C/C++ 应该使用 long long). 本题思路很直白,两重循环.外层 ...
- 删括号(dp)
题目链接:https://ac.nowcoder.com/acm/problem/21303 思路:删括号的时候一定要时刻保证左括号数量比右括号多,我们可以定义dp[i][j][k]表示考虑AA前i个 ...
- easyx学习心得
前几天算法课的实验要求实现可视化,搞了半天没动咋实现,然后有大佬说用easyx,,,我寻思着也没教这玩意咋用啊.然后很烦躁的上网找教程,发现没有教怎么使用的,都说有一本说明书(链接),自己调用函数就可 ...
- java随机数的产生
两种产生随机数的方法: 1.通过import java.util.Random来实现 2.Math.random() 一.第一种的话就是导入Random之后,先生成一个Random对象r,之后再利用r ...
- CodeCraft-20 (Div. 2) B. String Modification (字符串,规律)
题意:有一个长度为\(n\)的字符串,你可以选取一个值\(k(1\le k \le n)\),然后遍历字符串,每次将区间长度为\(k\)的字符串反转,求反转后字典序最小的字符串,并输出\(k\)的值. ...
- Educational Codeforces Round 91 (Rated for Div. 2) A. Three Indices (模拟)
题意:有一长度为\(n\)的序列,问是否能找到\(a_{i}<a_{j},a_{j}>a_{k},(i<j<k)\),如果满足,输出其位置. 题解:直接暴力两头找即可,最坏复杂 ...
- Win10 资源管理器经常卡死问题
什么问题? 我的Win10资源管理器开始出现卡死(假死.未响应)的情况,频率越来越高,触发方式越来越广,包括OpenDialog.打开拥有快捷方式的文件夹/桌面.右键空白处.右键文件/文件夹.拖动文件 ...
- Python+Selenium+Mysql(动态获取数据,数据库交互)
一.创建数据库连接 #!coding:utf-8 import pymysql ''' Python3之后不再支持MySQLdb的方式进行访问mysql数据库: 可以采用pymysql的方式 连接方式 ...