我需要在没有静态嵌套类的情况下实现Builder模式。有继承权的最佳方法是什么?
假设我有以下 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
}

最好创建一个Builder类来负责设置PassengerCar和Truck的值,还是我们需要3个附加类,CarBuilder,PassengerCarBuilder扩展CarBuilder和TruckBuilder扩展CarBuilder?

最佳答案

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

这是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();

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

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

渴望使用此构建器的版本应使用反射来创建要构建的对象:
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();

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

是否使用一个或另一个取决于您。 Plase让我知道它是否有错误,因为我无法对其进行测试,以及您是否有任何疑问(我已经习惯了这种代码,因此可能缺少一些有用的解释)。

07-27 13:33