读了《Effective Java》这本书,感触良多。它刷新了我对Java语言的认识,纠正了原来的许多编程坏习惯。感叹Java的世界原来是如此丰富,内功对于一名技术人员是多么重要!本文记录本书的第2条建议——用构建器(Builder)优雅、安全地构建对象。
在Java中有多种方式可以创建对象,总结起来主要有下面的4种方式:
- 正常创建。通过new 操作符
- 反射创建。调用Class或java.lang.reflect.Constructor的newInstance()方法
- 克隆创建。调用现有对象的clone()方法
- 发序列化。调用java.io.ObjectInputStream的getObject()方法反序列化
Java对象的创建方式是其语法明确规定,用户不可能从外部改变的。本文仍然要使用上面的方式来创建对象,所以本文只能说是构建对象,而非创建对象也。
假设有这样一个场景,现在要构建一个大型的对象,这个对象包含许多个参数的对象,有些参数有些是必填的,有些则是选填的。那么如何构建优雅、安全地构建这个对象呢?
单一构造函数
通常,我们第一反应能想到的就是单一构造函数方式。直接new的方式构建,通过构造函数来传递参数,见下面的代码:
/***
* 单一构造函数
*/
public class Person {
// 姓名(必填)
private String name;
// 年龄(必填)
private int age;
// 身高(选填)
private int height;
// 毕业学校(选填)
private String school;
// 爱好(选填)
private String hobby;
public Person(String name, int age, int height, String school, String hobby) {
this.name = name;
this.age = age;
this.height = height;
this.school = school;
this.hobby = hobby;
}
}
上面的构建方式有下面的缺点:
- 有些参数是可以选填的(如height, school),在构建Person的时候必须要传入可能并不需要的参数。
- 现在上面才5个参数,构造函数就已经非常长了。如果是20个参数,构造函数都可以直接上天了!
- 构建的这样的对象非常容易出错。客户端必须要对照Javadoc或者参数名来讲实参传入对应的位置。如果参数都是String类型的,一旦传错参数,编译是不会报错的,但是运行结果却是错误的。
多构造函数
对于第1个问题,我们可以通过构造函数重载来解决。见下面的代码:
/***
* 多构造函数
*/
public class Person {
// 姓名(必填)
private String name;
// 年龄(必填)
private int age;
// 身高(选填&