构建器之间的继承 - 如何处理类型?

Ily*_*gan 4 c# oop design-patterns builder

我在C#中有这个构建器(当然,这个例子很简单):

class BusBuilder
{
    Wheels mWheels = DefaultWheels;
    int mRoute = 0;

    public BusBuilder WithWheels(Wheels aWheels)
    {
        mWheels = aWheels;
        return this;
    }

    public BusBuilder WithRoute(int aRoute)
    {
        mRoute = aRoute;
        return this;
    }

    public Bus Build()
    {
        return new Bus { Wheels = mWheels, Route = mRoute };
    }
}
Run Code Online (Sandbox Code Playgroud)

它的使用方式如下:

Bus bus = 
    new BusBuilder()
    .WithWheels(someWheels)
    .WithRoute(50)
    .Build()
Run Code Online (Sandbox Code Playgroud)

现在我想提取一个只包含一些方法的超类:

class VehicleBuilder
{
    Wheels mWheels = DefaultWheels;

    public VehicleBuilder WithWheels(Wheels aWheels)
    {
        mWheels = aWheels;
        return this;
    }
}

class BusBuilder : VehicleBuilder
{
    ...
}
Run Code Online (Sandbox Code Playgroud)

问题是现在我不能写

Bus bus = 
    new BusBuilder()
    .WithWheels(someWheels)
    .WithRoute(50)
    .Build()
Run Code Online (Sandbox Code Playgroud)

因为WithWheels返回a VehicleBuilder而不是a BusBuilder,因此没有定义WithRoute方法.

你会怎么设计这个?

Jon*_*eet 13

在继承方面,构建器模式有点痛苦.你可以这样做:

class VehicleBuilder<T> where T : VehicleBuilder<T>
{
    private T @this;

    protected VehicleBuilder()
    {
        // Or pass it in as a constructor parameter
        @this = (T) this;
    }

    public T WithWheels(...)
    {
        return @this;
    }
}
Run Code Online (Sandbox Code Playgroud)

然后:

class BusBuilder : VehicleBuilder<BusBuilder>
{
    ...
}
Run Code Online (Sandbox Code Playgroud)

此时,您的WithWheels方法仍会返回,BusBuilder因此您仍然可以调用WithRoute.

你还需要Build在每个派生类构建器中使用一个新方法,请注意......