I am sure that most programmers managed to learn some specific design patterns before they even knew what design patterns were. This is certainly true for myself – and while I cannot say what my first design pattern was, I can tell you about the first one I remember learning: the Visitor pattern. I was midway through a university class on object oriented programming and in my spare time I was writing a turn based puzzle game in C# that involved daggers and goblins and dragons and doors and keys and whatnot. Now to make the thing work, I was doing something that felt very wrong. I was writing lots of switch statements that described the interactions between the different objects. I experimented, looking for better solutions using overloading and overriding, but none of it seemed to work. I asked my lecturer for help, and of course this is where I learned about visitor. Initially I was elated, but in time my opinion changed: visitor is this ugly bloated thing, designed to get around the lack of double dispatch support in most object oriented languages. Its a hack, a workaround. As with arguably all design patterns, its a trick to deal with limitations in the language.
So today I was watching this video that goes into more detail on the new C#4 features and something was said about runtime overload resolution that flicked a switch in my brain and suddenly I realised a fact that I am sure was mindblowingly obvious to people much smarter than I: the C#4 dynamic keyword supports double dispatch. If you are prepared to take the performance hit and just want to get something up and running, you can forget the visitor pattern. Lets look at an example based on the sort of thing I was trying to do years ago with my little game:
class Entity { public string Name { get; set; } public string GetGreeting(Entity e) { return "..."; } } class Foo : Entity { public string GetGreeting(Foo f) { return string.Format("Hi there {0}! Beware, there may be evil Bars nearby!", f.Name); } public string GetGreeting(Bar b) { return "Curse you! I hate all Bars!"; } } class Bar : Entity { public string GetGreeting(Bar b) { return string.Format("Greetings {0}! Have you seen any evil Foos lately?", b.Name); } public string GetGreeting(Foo f) { return "The Foos are a scourge on this planet! Take this!"; } } class StaticIntroducer { public void Introduce(Entity e1, Entity e2) { Console.WriteLine("{0} is introduced to {1} by the StaticIntroducer...", e1.Name, e2.Name); Console.WriteLine("{0} says: \"{1}\"", e1.Name, e1.GetGreeting(e2)); Console.WriteLine("{0} says: \"{1}\"", e2.Name, e2.GetGreeting(e1)); } } class Program { static void Main(string[] args) { Entity f1 = new Foo { Name = "BobFoo" }; Entity f2 = new Foo { Name = "JohnFoo" }; Entity b1 = new Bar { Name = "JerryBar" }; Entity b2 = new Bar { Name = "SamBar" }; var introducer = new StaticIntroducer(); introducer.Introduce(f1, f2); introducer.Introduce(f1, b1); introducer.Introduce(b1, b2); Console.ReadLine(); } }
It outputs the following:
BobFoo is introduced to JohnFoo by the StaticIntroducer... BobFoo says: "..." JohnFoo says: "..." BobFoo is introduced to JerryBar by the StaticIntroducer... BobFoo says: "..." JerryBar says: "..." JerryBar is introduced to SamBar by the StaticIntroducer... JerryBar says: "..." SamBar says: "..."
Clearly, it does not work as intended. The code I’ve written makes some sort of sense, but its not correct. If you look at the Introduce method, you can see that it takes two entities. As a result, the call to GetGreeting will be resolved at compile time to Entity.GetGreeting(Entity e) and no polymorphism will occur. You cannot fix the problem by simply making the methods virtual and overrides, as the overriden versions take different arguments. Redeclaring the GetGreeting methods using the new operator also does not help. If you are facing this problem with C# 3 or earlier, its time to apply the visitor pattern. I won’t do that here, there are plenty of examples of Visitor on the web.
But if you are using C#4, its time for dynamic to come to the rescue. Here is a slightly different implementation of an Introducer:
class DynamicIntroducer { public void Introduce(dynamic e1, dynamic e2) { Console.WriteLine("{0} is introduced to {1} by the DynamicIntroducer...", e1.Name, e2.Name); Console.WriteLine("{0} says: {1}", e1.Name, e1.GetGreeting(e2)); Console.WriteLine("{0} says: {1}", e2.Name, e2.GetGreeting(e1)); } }
The ONLY thing I have changed is the signature. The parameters are now typed as dynamic rather than Entity. I plug this introducer in and this is the output:
BobFoo is introduced to JohnFoo by the DynamicIntroducer... BobFoo says: Hi there JohnFoo! Beware, there may be evil Bars nearby! JohnFoo says: Hi there BobFoo! Beware, there may be evil Bars nearby! BobFoo is introduced to JerryBar by the DynamicIntroducer... BobFoo says: Curse you! I hate all Bars! JerryBar says: The Foos are a scourge on this planet! Take this! JerryBar is introduced to SamBar by the DynamicIntroducer... JerryBar says: Greetings SamBar! Have you seen any evil Foos lately? SamBar says: Greetings JerryBar! Have you seen any evil Foos lately?
It works! How? Well, once you are using the dynamic type, overload resolution occurs at runtime rather than compile time, and is done using the runtime types of the arguments rather than the compile time types. My nice, succinct code that describes the different interactions now works just how I intended. But at what cost? Putting the possible performance problems aside, we have made the code more fragile. If someone passes a non-Entity to the Introduce method, we’ll get a runtime error! Well this is actually a trivial fix that only occurred to me while writing this post:
public void Introduce(Entity e1, Entity e2) { Console.WriteLine("{0} is introduced to {1} by the DynamicIntroducer...", e1.Name, e2.Name); Console.WriteLine("{0} says: {1}", e1.Name, ((dynamic)e1).GetGreeting(e2)); Console.WriteLine("{0} says: {1}", e2.Name, ((dynamic)e2).GetGreeting(e1)); }
The only method call that actually needs to be dynamic is GetGreeting, so we can change the method signature back to as it was before and just introduce a cast to dynamic for the GetGreeting invocation. Much better. There is one more thing I would like to try to do: hide the need for GetGreeting to be invoked dynamically from the caller. The crude way is to use a static method, but I’m not a fan of that. Tomorrow I will see if there is an elegant way to approach it.
Good.
ReplyDeleteVisitor is made to separe algorithm from original object.
ReplyDeleteFor instance, you may have your entities and a visitor to play a sound matching the entity. You don't want a call to the sound library in your entity so you put it in a visitor.
Moreover your classes Foo and Bar have dependancies which is not always a good thing.
Sure. In this case the algorithm is the introducer. I don't follow your point about the Foo and Bar classes having dependencies though.
ReplyDelete