Why would I ever use a Chain of Responsibility over a Decorator?

后端 未结 11 1886
[愿得一人]
[愿得一人] 2020-12-04 12:58

I\'m just reading up on the Chain of Responsibility pattern and I\'m having trouble imagining a scenario when I would prefer its use over that of decorator.

What do

相关标签:
11条回答
  • 2020-12-04 13:17

    The fact that you can break the chain at any point differentiates the Chain of Responsibility pattern from the Decorator pattern. Decorators can be thought of as executing all at once without any interaction with the other decorators. Links in a chain can be thought of as executing one at a time, because they each depend on the previous link.

    Use the Chain of Responsibility pattern when you can conceptualize your program as a chain made up of links, where each link can either handle a request or pass it up the chain.

    When I used to work with the Win32 API, I would sometimes need to use the hooking functionality it provides. Hooking a Windows message roughly follows the Chain of Responsibility pattern. When you hooked a message such as WM_MOUSEMOVE, your callback function would be called. Think of the callback function as the last link in the chain. Each link in the chain can decide whether to throw away the WM_MOUSEMOVE message or pass it up the chain to the next link.

    If the Decorator pattern had been used in that example, you would have been notified of the WM_MOUSEMOVE message, but you would be powerless to prevent other hooks from handling it as well.

    Another place the Chain of Command pattern is used is in game engines. Again, you can hook engine functions, events, and other things. In the case of a game engine, you don't want to simply add functionality. You want to add functionality and prevent the game engine from performing its default action.

    0 讨论(0)
  • 2020-12-04 13:19

    The difference between these patterns is not related to when or how the chain can be broken (which assumes a chain) or in when the extra behaviour is executed. They are related in that they both use composition in favour of inheritance to provide a more flexible solution.

    The key difference is that a decorator adds new behaviour that in effect widens the original interface. It is similar to how normal extension can add methods except the "subclass" is only coupled by a reference which means that any "superclass" can be used.

    The COR pattern can modify an existing behaviour which is similar to overriding an existing method using inheritance. You can choose to call super.xxx() to continue up the "chain" or handle the message yourself.

    So the difference is subtle but an example of a decorator should help:

    interface Animal
    {
        Poo eat(Food food);
    }
    
    class WalkingAnimal implements Animal
    {
        Animal wrapped;
        WalkingAnimal(Animal wrapped)
        {
            this.wrapped = wrapped;
        }
    
        Position walk(Human walker)
        {
        };
    
        Poo eat(Food food)
        {
          return wrapped.eat(food);
        }
    }
    
    class BarkingAnimal implements Animal
    {
        Animal wrapped;
        BarkingAnimal(Animal wrapped)
        {
            this.wrapped = wrapped;
        }
    
        Noise bark()
        {
        };
    
        Poo eat(Food food)
        {
            bark();
            return wrapped.eat();
        }
    }
    

    You can see that we can compose a walking, barking animal... or in fact add the ability to bark to any animal. To use this extra behaviour directly we would need to keep a reference to the BarkingAnimal decorator.

    All BarkingAnimal's also bark once before eating which has changed existing functionality and so is similar to a COR. But the intent is not the same as COR i.e. to find one Animal of many that will eat the food. The intent here is to modify the behaviour.

    You could imagine a COR being applied to find a human that will take the animal for a walk. This could be implemented as a linked list like chained above or as an explicit List... or whatever.

    Hope this is reasonably clear!

    John

    0 讨论(0)
  • 2020-12-04 13:21

    I think the situations to apply these two patterns are different. And by the way, for decorator pattern, the decorator should know the component which it wrapped. And for CoR, the different interceptors could know nothing of each other.

    0 讨论(0)
  • 2020-12-04 13:23

    After reading the Gang of Four definitions, I'm not convinced there's a real difference. (included for convenience)

    • Decorator: Allows for the dynamic wrapping of objects in order to modify their existing responsibilities and behaviours
    • Chain of Responsibility: Gives more than one object an opportunity to handle a request by linking receiving objects together

    Wikipedia fleshes them out a little, but some of it's kinda arbitrary.

    • Decorator is typically implemented as a Linked List. But I think that's too low-level to be considered "part" of the pattern.
    • Chain of Responsibility links only handle data if it's their responsibility; but determining responsibility and data handling are both part of behavior. Decorators can do this just as easily.
    • Decorator requires you to call the delegate.
    • A "pure" CoR link should only call the delegate if it doesn't handle the data.

    The first two attributes don't really distinguish the patterns. The second two do, but the way Decorator and CoR are usually implemented don't enforce those attributes--the designer just hopes no one writes a Decorator that breaks the chain or a CoRLink that continues the chain after handling the data.

    To actually implement these attributes, you'd need something like the following.

    Enforced Decorator:

    abstract class Decorated {
    
    public Decorated delegate;
    
    public final Object doIt(Object args) {
        Object returnVal = behavior(arg);
        if(delegate != null) returnVal = delegate.doit(returnVal);
        return returnVal;
    }
    
    protected abstract Object behavior(Object args); //base or subclass behavior
    }
    

    Enforced Chain of Responsibility:

    abstract class Link {
    
    public Link delegate;
    
    public final Object processIt(Obect args) {
        Object returnVal = args;
        if(isMyResponsibility) returnVal = processingBehavior(returnVal);
        else returnVal = delegate.processIt(returnVal);
        return returnVal;
    }
    
    protected abstract Boolean isMyResponsibility(Object args);
    
    protected abstract Object processingBehavior(Object args);
    }
    

    (Alternately, you could just add a line to the javadoc, if all you want is to absolve yourself of the responsibiity in case someone else screws up your design--but why leave it to chance?)

    0 讨论(0)
  • 2020-12-04 13:30

    I agree that from structural standpoint this two patterns are very similar. My thought is about the final behavior:

    In the classic interpretation of CoR element which handles the request breaks the chain.

    If any element in decorator breaks the chain then it will be wrong implementation of decorator, because base part of behavior will be lost. And the idea of decorator is transparent addition of new behavior when the base behavior remains untouched.

    0 讨论(0)
  • 2020-12-04 13:31

    I'd say that a Chain of Responsibility is a particular form of Decorator.

    0 讨论(0)
提交回复
热议问题