本篇将拿构造器来和使用builder设计模式比较着解释应该如何创建不可变对象
使用此种方式可以使你的代码更具有可读性,首先,让我们来看下如果使用如下的接收三个BigDecimal类型参数的构造器来构建的CashBalance对象是多么的不优雅。然后我们会在看下使用builder模式是怎样让你的代码更简洁的。
当下次你在面试中被问及设计模式的时候,你可以拿这个实现来和使用很频繁的工厂模式和单例模式做个比较。这篇文章还设计到其他两个重要的概念(不可变性和线程安全)
import java.math.BigDecimal; /** * 因为不可变,所以是线程安全的对象 */ public final class CashBalance { private BigDecimal initialBalance; private BigDecimal totCredits; private BigDecimal totDebits; //构造器 public CashBalance(BigDecimal initialBalance, BigDecimal totCredits, BigDecimal totDebits) { this.initialBalance = initialBalance; this.totCredits = totCredits; this.totDebits = totDebits; } //不可变对象中只提供getter方法,不提供setter方法 }
因此,这段代码哪里不优雅?首先它的构造器需要三个BigDecimal的参数,在使用类的时候对于参数的意义来说不怎么直观,你说不清哪个参数是initialBalance,哪个参数是totCredits,如果你使用下面的方式来调用构造器会更好
CashBalance bal = new CashBalance(initialBalance:BigDecimal.valueOf(250.00), totCredits:BigDecimal.valueOf(250.00), totDebits:BigDecimal.valueOf(250.00));
很不幸的是,你不能使用上面的语法来写程序,你只能通过下面的方式调用它.
CashBalance bal = new CashBalance(BigDecimal.valueOf(250.00), BigDecimal.valueOf(250.00), BigDecimal.valueOf(250.00));
通过一个空构造器和三个setter方法来实现代码会更为优雅,但是对象必须是不可变的。下面是拯救你的构造器设计模式的代码。通过定义一个内部类来构造CashBalance对象。
import java.math.BigDecimal; /** * 不可变,所以线程安全 */ public final class CashBalance { private BigDecimal initialBalance, totCredits, totDebits; //构造器 public CashBalance(CashBalanceBuilder builder) { this.initialBalance = builder.initialBalance; this.totCredits = builder.totCredits; this.totDebits = builder.totDebits; } //构造器模式 public static class CashBalanceBuilder { // 有需要构造的对象一致的字段列表 protected BigDecimal initialBalance, totCredits, totDebits; //将访问权限设置为包内访问 void setInitialBalance(BigDecimal initialBalance) { this.initialBalance = initialBalance; } void setTotCredits(BigDecimal totCredits) { this.totCredits = totCredits; } void setTotDebits(BigDecimal totDebits) { this.totDebits = totDebits; } } //只提供setter方法 }
现在,你可以像下面调用一样在类外面构造CashBalance
public static void main(String[] args) { CashBalance.CashBalanceBuilder builder = new CashBalance.CashBalanceBuilder(); builder.setInitialBalance(BigDecimal.valueOf(250.00)); builder.setTotCredits(BigDecimal.valueOf(250.00)); builder.setTotDebits(BigDecimal.valueOf(250.00)); CashBalance bal = new CashBalance(builder); }
上面的代码完成了,但是如果你还有更多的字段,构造的代码就会显得很臃肿。这个可以用下面的方式来改善。
下面代码的改进是通过修改void类型的setter方法设值后返回builder自身来实现的
import java.math.BigDecimal; /** * 不可变线程安全对象 */ public final class CashBalance { private BigDecimal initialBalance, totCredits, totDebits; //构造器 public CashBalance(CashBalanceBuilder builder) { this.initialBalance = builder.initialBalance; this.totCredits = builder.totCredits; this.totDebits = builder.totDebits; } public static class CashBalanceBuilder { //has same fields as the object it is going to build protected BigDecimal initialBalance, totCredits, totDebits; //define the setters that return itself CashBalanceBuilder setInitialBalance(BigDecimal initialBalance) { this.initialBalance = initialBalance; return this; } CashBalanceBuilder setTotCredits(BigDecimal totCredits) { this.totCredits = totCredits; return this; } CashBalanceBuilder setTotDebits(BigDecimal totDebits) { this.totDebits = totDebits; return this; } } }
代码修改带来的优雅是现在可以通过下面的构造方式调用:
public static void main(String[] args) { CashBalance.CashBalanceBuilder builder = new CashBalance.CashBalanceBuilder() .setInitialBalance(BigDecimal.valueOf(250.00)) .setTotCredits(BigDecimal.valueOf(250.00)) .setTotDebits(BigDecimal.valueOf(250.00)); CashBalance bal = new CashBalance(builder); }
因此这些秘诀可以让你在和职场达人的面试中赢得加分