EffectiveJava笔记[一]
2019-07-21 本文已影响0人
pigrange
1、考虑使用静态工厂方法代替构造方法:
静态工厂方法的优点:
- 拥有名字,更加容易阅读
- 不需要每一次调用的时候都创建一个对象,可以对实例的数量进行严格的控制
- 可以返回其返回类型的任何子类型的对象
- 可以根据传入参数的不同而返回不同类型的对象,(构造方法只能返回自己的对象)
- 在编写包含该方法的类的时候,返回的对象的类不需要存在。
缺点:
- 如果只提供静态工厂方法,那么它就有可能没有public或者private的构造方法,也就是说无法被子类化。
- 没有一个统一的对静态构造方法的命名的规范,导致可能很难找到他们。
2、当构造方法的参数过多的时候,使用Builder模式:
静态工厂方法和普通的构造方法都无法很好的解决构造方法中参数很多且可选的参数很多的问题。
使用构造方法的缺点:
- 过多的方法重载,导致非常的繁琐
- 并且在获取类的实例的时候可选(参数)的构造方法太多,以至于降低可读性
1、JavaBean模式:
通过一个无参数(这里指可选参数)的构造方法来创建对象,然后调用一系列setter方法来设置参数
缺点:
- 代码较为冗余
- 由于是实现创建好了对象在设置值,所以在多线程的情况下无法保证对象参数的一致性(也就是说先创建好了对象再去设置参数,无法保证在创建过程中对象的不可变的)
2、Builder模式:
客户端不直接创建对象,而是通过一个builder预先配置好对象的每一个可选参数,然后通过builder的build方法来生成对象,这样产生的对象就是不可改变的,Builder通常是它所构建的类的一个静态的成员。
3、使用私有构造方法或枚举来实现单例
单例通常表示的是无状态的对象。比如说一个只提供函数功能的对象或者一个系统组件。
1、实现单例的方式:
- 创建一个静态final的自身引用,并私有构造方法
- 提供一个公共静态方法用于返回自身的单例。
- 注意,对于构造方法,应当在请求创建第二个实例的时候抛出异常,因为Java的反射机制可以调用private的构造方法。
2、单例与序列化:
如果单例的对象需要支持序列化的话,那么仅仅将这个类直接实现Serializable接口是不行的,这个问题正好是前面应该注意的点的良好反应。
当使用ObjectInuptStream对对象进行反序列化的时候,Java会通过反射去调用类的无参构造方法(尽管它是私有的),当然如果按照我们前面的实现,那么很有可能在反序列化的时候直接抛出了异常(因为请求创建了第二个实例)
解决方案:
- 在单例类中提供一个readResolve方法,并在这个方法中返回我们的单例对象。
4、使用私有构造方法来执行非实例化
这个条目是直接基于EffectiveJava直译而来。通俗来讲就是对于那些只用于提供静态方法或者静态属性的类(比如某些Util工具类或者config配置文件),我们应该避免他们被创建实例或者继承,因为并没有任何意义。
采取的操作:
- 将类设置为final,避免被继承
- 将类的构造方法设置为私有,并在代码块里面抛出异常(防止别人通过反射来创建实
5、依赖注入优于硬连接资源
这个又是显得异常的高大上。至少对于我来说,看见依赖注入就觉得这个是高大上且较难的东西,并且第一遍读这个条目的时候也没有搞明白这个到底是啥。
其实弄明白了什么是依赖注入,这个条目就非常的简单了。
依赖注入:
- 依赖:举个比较普遍的例子。我把我自己比作对象a,我的电脑比作对象b。如果我需要写这篇blog,我就需要我的电脑,也就是说a依赖于b。
- 依赖注入:对于a依赖的对象b,我们不应该在a的类中直接创建b的实例,而是提供b的设置api或者通过构造方法将b的引用传入。也就是说将a的依赖b注入给了b。
- 这样做有两个好处,一是b的生命周期并不用需要a来进行管理。二是对于多个依赖b的对象,如果b是不可变的,那么可以实现b的复用。
- 虽然通过接口传递依赖也是一种实现,不过不推荐这么做,因为在并发的环境中,这是非常容易出错的。除非我们的对象非得支持多种依赖,那么最好的解决方式还是将其设置为final然后再通过构造方法传入。
依赖注入的有用变体:
将资源工厂传递给构造方法。