Builder は Cloneable を実装し、clone() をオーバーライドします。ビルダーのすべてのフィールドをコピーする代わりに、不変クラスはビルダーのプライベート クローンを保持します。これにより、新しいビルダーを返し、不変インスタンスのわずかに変更されたコピーを簡単に作成できます。
これで行けます
MyImmutable i1 = new MyImmutable.Builder().foo(1).bar(2).build();
MyImmutable i2 = i1.builder().foo(3).build();
Cloneable インターフェースは多少壊れていると言われていますが、これは適切な Java コーディング慣行に違反していませんか? この構造に問題はありますか?
final class MyImmutable {
public int foo() { return builder.foo; }
public int bar() { return builder.bar; }
public Builder builder() { return builder.clone(); }
public static final class Builder implements Cloneable {
public Builder foo(int val) { foo = val; return this; }
public Builder bar(int val) { bar = val; return this; }
public MyImmutable build() { return new MyImmutable(this.clone()); }
private int foo = 0;
private int bar = 0;
@Override public Builder clone() { try { return (Builder)super.clone(); } catch(CloneNotSupportedException e) { throw new AssertionError(); } }
}
private MyImmutable(Builder builder) { this.builder = builder; }
private final Builder builder;
}