在 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;
// 身高(选填)
private int height;
// 毕业学校(选填)
private String school;
// 爱好(选填)
private String hobby; public Person(String name, int age) { this.name = name; this.age = age; } public Person(String name, int age, int height) { this.name = name; this.age = age; this.height = height; } public Person(String name, int age, int height, String school) {this.name = name; this.age = age; this.height = height; this.school = school; } public Person(String name, int age, String hobby, String school) { this.name = name; this.age = age; this.hobby = hobby; this.school = school; }}
上面的方式确实能在一定程度上降低构造函数的长度,但是却有下面的缺陷:
导致类过长。这种方式会使得 Person 类的构造函数成阶乘级增长。按理来说,应该要写的
构造函数数是可选成员变量的组合数(实际并没有这么多,原因见第 2 点)。如果让我调用
这样的类,绝对会在心里默念 xx!!
有些参数组合无法重构。因为 Java 中重载是有限制的,相同方法签名的方法不能构成重
载,编译时无法通过。譬如包含(name, age, school)和(name, age, hobby)的构造函数是不
能重载的,因为 shcool 和 hobby 同为 String 类型。Java 只认变量的类型,管你变量是什么
含义呢。 (看脸的社会唉)
JavaBean 方式
上面的方法不行,莫急!还有法宝——JavaBean。一个对象的构建通过多个方法来完成。
直接见下面的代码:
public class Person {
// 姓名(必填)
private String name;
// 年龄(必填)
private int age;
// 身高(选填)
private int height;
// 毕业学校(选填)
private String school;
// 爱好(选填)
private String hobby; public Person(String name, int age) {this.name = name; this.age = age; } public void setHeight(int height) { this.height = height; } public void setSchool(String school) { this.school = school; } public void setHobby(String hobby) { this.hobby = hobby; }}
客户端使用这个对象的代码如下:
public class Client { public static void main(String[] args) { Person person = new Person("james