Home
  Latest posts
  My Writings
  My Code
  My Gallery
  About me
 
  rssfeed Syndication
 
Bloggtoppen.se
 
 
Links
  Cornerstone
  SweNug
 
Post categories
  misc (48)
  Architecture (21)
  C# (19)
  Asp.Net (2)
  Vb.Net (2)
  Training (7)
  Data (19)
  Events (40)
  Platform (2)
  Orcas (4)
  Updates (3)
  Methods (10)
  Tools (6)
  Announcements (14)
  Languages (1)
  Patterns (6)
  Opinions (11)
  Fun (3)
  Ineta (1)
  Opinion (0)
  Practices (2)
  WCF (5)
 
 
 

Pattern Focus: Strategy Pattern using delegates

Friday, February 09, 2007

Interfaces are the classical way of implementing the strategy pattern, but with .NET we get another similar approach to technically accomplish the same thing. That is by utilizing delegates as part of our parameters list and C# anonymous delegates as our vessel for the strategy.

The first ever discussion I had with Erik Dörnenburg  (http://www.doernenburg.com/) was about delegates, he categorized them as a template interface with one method. While this is not true to a full extent, since they are more complex than that, it is a nice way to look at them in the context of strategy.

Changing the implementation
Instead of creating your own interface and a specific implementation of that interface, you can let the delegate plumbing do it for you. Continuing from the previous example, we could use delegates as the common denominator and act as an interface for the strategies we want. To do this, we just need to create a delegate with the appropriate signature, similar to the signature of the one single method the interface we created in the previous example had.

public delegate bool StrategyDelegate(Consultant consultant); 

Since this delegate will act as our interface, we need to change the parameter list to accept an instance of that delegate instead of the something implementing the IStrategy interface. This will allow us to pass any method with the same signature as the delegate as an in parameter to the method.

Consultant FindFirstSuitable(List consultants, StrategyDelegate strategy)

And finally we need to change the code in the foreach loop slightly:

foreach (Consultant consultant in consultants)  {
                if (strategy.Invoke(consultant))
                    return consultant;
            }

This accomplishes the exact same functionality as we did using interfaces. Although the intent is slightly less clear inside the method for this approach (Invoke don not tell me that much about what it actually checks), I will be able to utilize this function without creating new classes and objects.

Also; with C# 2.0 and the ability to create delegates on the fly, so called anonymous delegates, I will be able to be very precise and clear about the intent when I call this method. For example if I wanted to find a cheap consultant I could write:

Consultant consultant = FindFirstSuitable(consultants,   
                                          delegate(Consultant consultant)
                                          { return consultant.HourRate < 20;
                                          );

When I read this method call; it is quite obvious what my intent is and what the rule passed in actually stipulates. I do not have to look for details in external methods or type definitions.

Looking ahead and into the next version of C# (3.0 or whatever number it will get) this will be even neater with some Lambda expressions from the LINQ package. With Lambda you can express an anonymous delegate like this:

 Consultant consultant = FindFirstSuitable(consultants, consultant => consultant.HourRate <20);

Now that is just straight up beautiful!

Strategy pattern usage in the .Net Framework
This usage of the strategy pattern can be found in many places of the Framework but the most evident is in the List<> class. I have written a sum-up earlier on how that is used so I will not go into details but if you are interested you can read about it here:  https://lowendahl.net/showShout.aspx?id=86

Conclusions
When do we use interfaces and when do we use delegates? There is no clear rule but some guidelines. First of all delegates are much harder to reuse, especially if you use anonymous delegates since they are impossible to reuse, but they are very clear and concise in their intent. Interfaces on the other hand is a bit more complex to implement but will enable you to reuse the strategies across your domain and will let you have more than one method that might be used as part of the decision making. Personally I think that of the two solutions we can use today, interfaces are the more beautiful one (but that will probably change when Lambda is implemented).

The example code for this approach is attached to this post, rss readers need to go to the website to get it.

Attached file: StrategyPattern_Delegates.zip
 

Comments
2/19/2007 9:13:00 PM   http://vidarpactica.com.ar   -   NicoGranelli
 
To me, this is more like specification pattern than a strategy.
 
2/19/2007 10:17:00 PM   https://lowendahl.net   -   Patrik Löwendahl
 
Yes you are right but as I see it, specification is just a more narrow implementation of the more generic Strategy pattern. But that is only my point of view, there has been an ongoing debate in the DDD community about the relationship between the two. The problem with them is that depending on the complexity of the algorithms they look more or less the same. If I in my example had changed it slightly and had done some calculations, different calculations that either took level in account or not for example, it would have been a much clearer case of Strategy.

The basis for the discussion in the DDD community is just this, if the intent of the algorithm is important enough to not group specification together with strategy.
 
12/15/2009 8:14:00 AM   http://www.visual-guard.com/   -   Application Security
 
Nice Post..
Thanks For sharing the information..
 
4/1/2010 5:32:00 PM   http://blogs.dotnetfreaks.net   -   Ajander
 
good article


Comment
Title:
Your name:
Your url:
Text:
Please enter the text from the image: