建立对象不优雅?Java大牛手把手教你如何建立优雅的对象

建立对象不优雅?Java大牛手把手教你如何建立优雅的对象

 

在 Java 中有多种方式能够建立对象,总结起来主要有下面的 4 种方式:java

正常建立。经过 new 操做符程序员

反射建立。调用 Class 或 java.lang.reflect.Constructor 的 newInstance()方法数组

克隆建立。调用现有对象的 clone()方法安全

发序列化。调用 java.io.ObjectInputStream 的 getObject()方法反序列化多线程

Java 对象的建立方式是其语法明确规定,用户不可能从外部改变的。本文仍然要使用上面函数

的方式来建立对象,因此本文只能说是构建对象,而非建立对象也。ui

假设有这样一个场景,如今要构建一个大型的对象,这个对象包含许多个参数的对象,有this

些参数有些是必填的,有些则是选填的。那么如何构建优雅、安全地构建这个对象呢?.net

单一构造函数线程

一般,咱们第一反应能想到的就是单一构造函数方式。直接 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 只认变量的类型,管你变量是什么

含义呢。 (看脸的社会唉)

建立对象不优雅?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", 12);
 person.setHeight(170);
 person.setHobby("reading");
 person.setSchool("xxx university");
 }
}

这样看起来完美的解决了 Person 对象构建的问题,使用起来很是优雅便捷。确实,在单一

线程的环境中这确实是一个很是好的构建对象的方法,可是若是是在多线程环境中仍有其

致命缺陷。在多线程环境中,这个对象不能安全地被构建,由于它不是不可变对象。一旦

Person 对象被构建,咱们随时可经过 setXXX()方法改变对象的内部状态。假设有一个线程

正在执行与 Person 对象相关的业务方法,另一个线程改变了其内部状态,这样获得莫名

其妙的结果。因为线程运行的无规律性,使得这问题有可能不能重现,这个时候真的就只

能哭了。(程序员真苦逼。。。)

Builder 方式

为了完美地解决这个问题,下面引出本文中的主角(等等等等!)。咱们使用构建器

(Builder)来优雅、安全地构建 Person 对象。废话不说,直接代码:

/**

* 待构建的对象。该对象的特色:

* <ol>

* <li>须要用户手动的传入多个参数,而且有多个参数是可选的、顺序随意</li>

* <li>该对象是不可变的(所谓不可变,就是指对象一旦建立完成,其内部状态不可变,

更通俗的说是其成员变量不可改变)。

* 不可变对象本质上是线程安全的。</li>

* <li>对象所属的类不是为了继承而设计的。</li>

* </ol>

* 知足上面特色的对象的构建但是使用下面的 Build 方式构建。这样构建对象有下面的好

处:

* <ol>

* <li>不须要写多个构造函数,使得对象的建立更加便捷</li>

* <li>建立对象的过程是线程安全的</li>

* </ol>
* @author xialei
* @date 2015-5-2
*/
public class Person {

// 姓名(必填),final 修饰 name 一旦被初始化就不能再改变,保证了对象的不可变

性。

private final String name;

// 年龄(必填)

private final int age;

// 身高(选填)

private final int height;

// 毕业学校(选填)

private final String school;

// 爱好(选填)

private final String hobby;

/**

* 这个私有构造函数的做用:

* <ol>

* <li>成员变量的初始化。final 类型的变量必须进行初始化,不然没法编译成功</li>

* <li>私有构造函数可以保证该对象没法从外部建立,而且 Person 类没法被继承</li>

* </ol>
 */
 private 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;
}
 /**

* 要执行的动做

*/
 public void doSomething() {
 // TODO do what you want!!
 }
 /**

* 构建器。为何 Builder 是内部静态类?

* <ol>

* <li>必须是 Person 的内部类。不然,因为 Person 的构造函数私有,不能经过 new 的

方式建立 Person 对象</li>

* <li>必须是静态类。因为 Person 对象没法从外部建立,若是不是静态类,则外部无

法引用 Builder 对象。</li>

* </ol>

* <b>注意</b>:Builder 内部成员变量要与 Person 的成员变量保持一致。

* @author xialei

*

*/

public static class Builder {

// 姓名(必填)。注意:这里不能是 final 的

private String name;

// 年龄(必填)

private int age;

// 身高(选填)

private int height;

// 毕业学校(选填)

private String school;

// 爱好(选填)

private String hobby;
 public Builder(String name, int age) {
 this.name = name;
 this.age = age;
 }
 public Builder setHeight(int height) {
 this.height = height;
return this;
 }
 public Builder setSchool(String school) {
 this.school = school;
 return this;
 }
 public Builder setHobby(String hobby) {
 this.hobby = hobby;
 return this;
 }
 /**

* 构建对象

* @return 返回待构建的对象自己

*/
 public Person build() {
 return new Person(name, age, height, school, hobby);
 }
 }
}

客户端构建对象的方式见下面的代码:

/**

* 使用 Person 对象的客户端

* @author xialei
* @date 2015-5-2
*/
public class Client {
 public static void main(String[] args) {
 /*

* 经过链式调用的方式建立 Person 对象,很是优雅!

*/
 Person person = new Person.Builder("james", 12)
 .setHeight(170)
 .setHobby("reading")
 .build();
 person.doSomething();
 }
}

若是不想看代码,可看下面对于上面代码的总结:

经过 private Person(..)使得 Person 类不可被继承

经过将 Person 类的成员变量设置为 final 类型,使得其不可变

经过 Person 内部的 static Builder 类来构建 Person 对象

经过将 Builder 类内部的 setXXX()方法返回 Builder 类型自己,实现链式调用构建 Person 对

总结

至此,咱们就相对完美地解决这一类型的对象建立问题!下面来总结一下本文的重点。待

建立的对象特色:

须要用户手动的传入多个参数,而且有多个参数是可选的、顺序任意

对象不可变

对象所属的类不是为了继承而设计的。即类不能被继承

依次使用的对象构建方法:

单一构造函数

多构造函数

JavaBean 方式

Builder 方式

最终,经过比较得出 Builder 方法最为合适的解决。

相关文章
相关标签/搜索