Effective Java
创建和销毁对象
- 用静态工厂方法代替构造器
- 遇到多个构造器参数时要考虑使用构建器
- 用私有构造器或者枚举类型强化 Singleton 属性
- 通过私有构造器强化不可实例化的能力
- 优先考虑依赖注入来引入资源
- 避免创建不必要的对象
- 清除过期的对象的引用
- 避免使用终结方法和清除方法
- try-with-resources 优先于 try-finally
对于所有对象都通用的方法
- 覆盖 equals 时请遵守通用约定
- 覆盖 equals 时总要覆盖 hashCode
- 始终要覆盖 toString
- 谨慎地覆盖 clone
- 考虑实现 Comparable 接口
类和接口
- 使用类和成员的可访问性最小化
- 要在公有类中使用访问方法而非公有域
- 使可变性最小化
- 复合优先于集成
- 要么设计继承并提供文档说明,要么禁止继承
- 接口优于抽象类
- 为后代设计接口
- 接口只用于定义类型
- 类层次优于标签类
- 静态成员类优于非静态成员类
- 限制源文件为单个顶级类
泛型
- 请不要使用原生态类型
- 清除非受检的警告
- 列表优于数组
- 优先考虑泛型方法
- 利用有限制通配符来提升 API 的灵活性
- 谨慎并用泛型和可变参数
- 优先考虑类型安全的异构容器
枚举和注解
- 用 enum 代替 int 常量
- 用实例域代替序数
- 用 EnumSet 代替位域
- 用 EnumMap 代替序数索引
- 用接口模拟可扩展的枚举
- 注解优于命名模式
- 坚持使用 Override 注解
- 用标记接口定义类型
Lambda 和 Stream
- Lambda 优先于匿名类
- 方法引用优先于 Lambda
- 坚持使用标准的函数接口
- 谨慎使用 Stream
- 优先选择 Stream 中无副作用的函数
- Stream 要优先用 Collection 作为返回类型
- 谨慎使用 Stream 并行
方法
- 检查参数的有效性
- 必要时进行保护性拷贝
- 谨慎设计方法签名
- 慎用重载
- 慎用可变参数
- 返回零长度的数据或者集合,而不是 null
- 谨慎返回 optional
- 为所有导出的 API 元素编写文档注释
通用编程
- 将局部变量的作用域最小化
- for-each 循环优先于传统的 for 循环
- 了解和使用类库
- 如果需要精确的答案,请避免使用 float 和 double
- 基本类型优于装箱基本类型
- 如果其他类型更适合,则尽量避免使用字符串
- 了解字符串连接的性能
- 通过接口引用对象
- 接口优于反射机制
- 谨慎地使用本地方法
- 谨慎地进行优化
- 遵守普遍接受的命名惯例
异常
- 只针对异常的情况才使用异常
- 对可恢复的情况使用受检异常,对编程错误使用运行时异常
- 避免不必要地使用受检异常
- 优先使用标准的异常
- 抛出与抽象对象对应的异常
- 每个方法抛出的所有异常都要建立文档
- 在细节消息中包含失败-捕获信息
- 努力使失败保持原子性
- 不要忽略异常
并发
- 同步访问共享的可变数据
- 避免过度同步
- executor、task 和 stream 优先于线程
- 并发工具优先于 wait 和 notify
- 线程安全性的文档化
- 慎用延迟初始化
- 不要依赖线程调度器
序列化
- 其他方法优先于 Java 序列化
- 谨慎地实现 Serializable 接口
- 考虑使用自定义的序列化形式
- 保护性地编写 readObject 方法
- 对于实例控制,枚举类型优先于 readResolve
- 考虑用序列化代理代替序列化实例