I am revising the Visitor pattern I used some time ago. We have base class Element which has virtual method accept(Visitor) and this method is overridden in all classes inheriting from Element. And all that accept() does in any derived class is to call visitor->visit(*this). Now when the client runs the code, he/she does for example:
Visitor& theVisitor = *new ConcreteVisitor();
for_each(elements.begin(), elements.end(), [](Element& e) { e.accept(theVisitor));})
Why the client cannot just call visitor->visit(element) like this:
Visitor& theVisitor = *new ConcreteVisitor();
for_each(elements.begin(), elements.end(), [&theVisitor](Element& e) { theVisitor.visit(e); });
What useful information is in calling element.accept(visitor) which in turns calls visitor.visit(element)? This makes the usage of the Visitor pattern cumbersome and requires extra code in all the hierarchy of Element classes.
So can someone explain benefits of accept() here?
I've been long confused with the Visitor pattern and I've been trying to find explanations all over the Internet and these explanations confused me even more. Now I realised the reasons why Visitor pattern is needed and the way how it is implemented, so here it is:
Visitor pattern in needed to solve the Double Dispatch problem.
Single dispatch - when you have one class hierarchy and you have an instance of a concrete class in this hierarchy and you want to call an appropriate method for this instance. This is solved with function overriding (using virtual function table in C++).
Double dispatch is when you have two class hierarchies and you have one instance of concrete class from one hierarchy and one instance of concrete class from the other hierarchy and you want to call the appropriate method which will do the job for those two particular instances.
Let's look at an example.
First class hierarchy: animals. Base:
Animal
, derived:Fish
,Mammal
,Bird
. Second class hierarchy: invokers. Base:Invoker
, derived:MovementInvoker
(move the animal),VoiceInvoker
(makes the animal to sound),FeedingInvoker
(feeds the animal).Now for every specific animal and every specific invoker we want just one specific function to be called that will do the specific job (e.g. Feed the Bird or Sound the Fish). So altogether we have 3x3 = 9 functions to do the jobs.
Another important thing: the client who runs each of those 9 functions does not want to know what concrete
Animal
and what concreteInvoker
he or she has got at hand.So the client wants to do something like:
Or:
Now: how is it possible in RUN-TIME to call one of the 9 (or whatever) specific methods that deals with this specific
Animal
and this specificInvoker
?Here comes Double Dispatch. You absolutely need to call one virtual function from first class hierarchy and one virtual function from the second.
So you need to call a virtual method of
Animal
(using virtual function table this will find the concrete function of the concrete instance in theAnimal
class hierarchy) and you also need to call a virtual method ofInvoker
(which will find the concrete invoker).YOU MUST CALL TWO VIRTUAL METHODS.
So here is the implementation (you can copy and run, I tested it with g++ compiler):
visitor.h:
visitor.cpp:
Output of the above code:
So what happens when the client has got an instance of
Animal
and an instance ofInvoker
and callsanimal.accept(invoker)
?Suppose that the instance of
Animal
isBird
and the instance ofInvoker
isFeedingInvoker
.Then thanks to virtual function table
Bird::accept(Invoker&)
will be called which will in turn runinvoker.doTheJob(Bird&)
. AsInvoker
instance isFeedingInvoker
, virtual function table will useFeedingInvoker::accept(Bird&)
for this call.So we made the double dispatch and called the correct method (one of 9 possible methods) for
Bird
andFeedingInvoker
.Why is Visitor pattern good?
The client does not need to depend on both complex class hierarchies of Animals and Invokers.
If new concrete animal (say,
Insect
) needs to be added, no existingAnimal
hierarchy needs to be changed. We only need to add:doTheJob(Insect& insect)
toInvoker
and all derived invokers.Visitor pattern elegantly implements the open/closed principle of object-oriented design: the system should be open to extensions and closed to modifications.
(In classic Visitor pattern
Invoker
would be replace byVisitor
anddoTheJob()
byvisit()
, but to me these names don't actually reflect the fact that some job is done on elements).