开发者

Adding a bifurcation to a fluent interface builder

开发者 https://www.devze.com 2023-02-01 23:31 出处:网络
I have a small framework that allows me to create a pipe&filter system. I had the idea of using fluent interfaces to build the pipe&filter system:开发者_StackOverflow中文版

I have a small framework that allows me to create a pipe&filter system. I had the idea of using fluent interfaces to build the pipe&filter system:开发者_StackOverflow中文版

        PipeFilter pipeFilter = PipeFilter.StartBuild()
            .AddFilter(new SomeFilter1())
            .AddFilter(new SomeFilter2())
            .AddFilter(new SomeFilter3())
            .AddFilter(new SomeFilter4())
            .Build();

The shown code works as expected. Here is a "picture" of the system:

SomeFilter1 -> SomeFilter2 -> SomeFilter3 -> SomeFilter4

Now, there is a kind of Filter that instead of one output has two, instead. I call it bifurcation. Here is an example of a system with a bifurcation:

              |-> SomeFilter2 -> SomeFilter3
SomeFilter1 --|
              |-> SomeFilter4

I'd like to implement something like this:

        PipeFilter pipeFilter = PipeFilter.StartBuild()
            .AddFilter(new SomeFilter1())
            .AddBifurcation()
                .Output1()
                    .AddFilter(new SomeFilter2())
                    .AddFilter(new SomeFilter3())
                .Output2()
                    .AddFilter(new SomeFilter4())
            .Build();

But it seems I just can't get it right. Is this even possible, to do? In the first example, I just needed a PipeFilterBuilder (that is being returned by PipeFilter.StartBuild()). In this second example, I've tried creating other kinds of builders to bring into the mix but that seems to be of no avail.

Forgot to mention, the idea would be that I could nest bifurcations anywhere I want, that is, I could get "trees" full of branches!

Can anybody be of any help with this?


I would go the following way

PipeFilter pipeFilter = PipeFilter.StartBuild()
        .AddFilter(new SomeFilter1())
        .AddBifurcation(
              withOutput(1)
                 .AddFilter(new SomeFilter2())
                 .AddFilter(new SomeFilter3()), /* this separates first and second output */
              withOutput(2)
                .AddFilter(new SomeFilter4())
              )
        .Build();

In more format terms, I define the Bifurcation class to be an implementor of the Filter interface.

A bifurcation can have any number of Output filters, chained using Output object. To distinguish these output objects, they all have an index.

As a consequence, addBifurcation create a new Bifurcation object and adds it, while withOutput(int) is a static method creating an Output object, which has all the required method. Notice this point implies you get rid of the classical distinction between Builder and builded object, in favor of a code where the Builder methods are defined in the base interface for Filter.


I think you are being constrained by your notation. At the lowest level, your filtering system can be composed of primitive filters, and sequential and parallel compositions. Your first example could be written (in pseudocode):

pipeFilter = Seq(new SomeFilter1(), 
                 Seq(new SomeFilter2(), 
                     Seq(new SomeFilter3(), new SomeFilter4())));

With such an interface it is completely obvious how to add parallel -- or really any other kind of combinator -- into the interface:

pipeFilter = Seq(new SomeFilter1(), 
                Parallel(Seq(new SomeFilter2(), new SomeFilter3()),
                         Seq(new SomeFitler4())));

Cumbersome though it may seem, I would suggest building your interface this way (called a "functional" as oppposed to "imperative" interface), and then writing convenience methods to reduce some of the structural burden, for example variants of Seq and Parallel that take an arbitrary numbers of arguments -- but it would probably be best to have those simply delegate down to folds of the binary variants.

To elaborate on the subtler design issue here, the class or interface with which you are working is a filter builder, not a filter itself. Parallel and Seq are methods on that class. This gives you the option to implement the combinators in multiple ways for multiple interpretations. I would write the interface like this:

interface FilterBuilder<Filter> {
    Filter Seq(Filter a, Filter b);
    Filter Parallel(Filter a, Filter b);
}

It may not be perfect for your needs, but it is a nice, flexible design pattern that doesn't seem to be widely known.


It is possible to implement the system as you've designed it.

You won't need any builder other than PipeFilterBuilder, but you will need a data structure capable of representing your tree of Filters. Your PipeFilterBuilder then holds a reference to this structure and keeps track of the current insertion point.

Any operation you perform on PipeFilterBuilder needs to update the insertion point and return the builder itself (this). Calling AddBifurcation would add the current insertion point to, say, a stack. Conversely, Output2 would set the insertion point to the value popped off the stack. Other functions should be fairly trivial.

0

精彩评论

暂无评论...
验证码 换一张
取 消