gpt4 book ai didi

java - builder 模式

转载 作者:塔克拉玛干 更新时间:2023-11-02 19:21:21 25 4
gpt4 key购买 nike

我需要在没有静态嵌套类的情况下实现构建器模式。如果我有继承权,最好的方法是什么?假设我有以下类(class)。

public class Car {
private String brand;
private String speed;
//getters an setters
}

public class PassengerCar extends Car{
private String capacity;
//getters an setters
}

public class Truck extends Car{
private String length;
//getters an setters
}

是创建一个负责设置 PassengerCar 和 Truck 值的 Builder 类更好,还是我们需要 3 个额外的类,CarBuilder、PassengerCarBuilder 扩展 CarBuilder 和 TruckBuilder 扩展 CarBuilder?

最佳答案

正确的方法是每个类一个生成器。我看到了两种不同的构建器实现,让我们称它们为lazyeager(也许其中一个不是严格的构建器,但它们实际上都是构建 个实例)。

这里是 CarTruck 的惰性构建器:

public abstract class AbstractLazyCarBuilder<T extends Car, B extends AbstractLazyCarBuilder<T, B>> {

private String brand;

private String speed;

public B brand(String brand) {
this.brand = brand;
return (B) this;
}

public B speed(String speed) {
this.speed = speed;
return (B) this;
}

public T build() {
T car = this.create();
this.fill(car);
return car;
}

protected abstract T create();

protected void fill(T car) {
car.setBrand(this.brand);
car.setSpeed(this.speed);
}
}

public class LazyCarBuilder extends AbstractLazyCarBuilder<Car, LazyCarBuilder> {

@Override
protected Car create() {
return new Car();
}
}

public class LazyTruckBuilder extends AbstractLazyCarBuilder<Truck, LazyTruckBuilder> {

private String length;

public LazyTruckBuilder length(String length) {
this.length = length;
return this;
}

@Override
protected Truck create() {
return new Truck();
}

@Override
protected void fill(Truck truck) {
super.fill(truck); // very important! fills truck with car's common attributes
truck.setLength(this.length);
}
}

用法:

Truck truck = new LazyTruckBuilder().brand("ford").speed("40").length("30").build();

它可能不是标准的构建器实现。它有两个通用参数:正在构建的对象的类型和构建器本身的类型。最后一个是为了避免在使用构建器方法时强制转换返回的对象。

它有一个返回空特定实例的create()方法(也可以通过反射创建)和一个设置所有属性的fill()方法到创建的对象。这个构建器是惰性的,因为对象是在您调用构建器上的 build() 时创建和初始化的。

这个构建器的 eager 版本应该使用反射来创建正在构建的对象:

public abstract class AbstractEagerCarBuilder<T extends Car, B extends AbstractEagerCarBuilder<T, B>> {

protected final T instance; // needs to be seen by subclasses

protected AbstractEagerCarBuilder() {
try {
// Reflection magic to get type of specific car
ParameterizedType type = (ParameterizedType) this.getClass().getGenericSuperclass();
Class<T> clazz = (Class<T>) type.getActualTypeArguments()[0];
// Create the specific car by reflection
this.instance = clazz.getConstructor().newInstance();
} catch (Exception e) {
throw new RuntimeException("Could not create specific instance", e);
}
}

public B brand(String brand) {
this.instance.setBrand(brand);
return (B) this;
}

public B speed(String speed) {
this.instance.setSpeed(speed);
return (B) this;
}

public T build() {
return this.instance;
}
}

public class EagerCarBuilder extends AbstractEagerCarBuilder<Car, EagerCarBuilder> {
// empty: just pass generic parameters
}

public class EagerTruckBuilder extends AbstractEagerCarBuilder<Truck, EagerTruckBuilder> {

private String length;

public EagerTruckBuilder length(String length) {
this.instance.setLength = length;
return this;
}
}

用法:

Truck truck = new EagerTruckBuilder().brand("gmc").speed("45").length("32").build();

这里,卡车实例实际上是在构建器创建时创建的。然后,构建器方法用属性一个一个地填充急切创建的实例。

是否使用其中一个,由您决定。如果这有错误,请告诉我,因为我无法测试它,如果您有任何问题(我已经习惯了这种代码,可能缺少一些有用的解释)。

关于java - builder 模式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28645730/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com