Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Difference betwen Visitor pattern & Double Dispatch

I am reading about visitor pattern, and it appears the same like Double Dispatch. Is there any difference between the two. Do the two terms means the same thing.

reference: http://www.vincehuston.org/dp/visitor.html

like image 826
vamsi Avatar asked Mar 22 '12 07:03

vamsi


People also ask

What does visitor pattern do?

The purpose of a Visitor pattern is to define a new operation without introducing the modifications to an existing object structure. Imagine that we have a composite object which consists of components.

When should you use the visitor pattern?

2 Answers. Show activity on this post. The visitor pattern is useful when you want to process a data structure containing different kinds of objects, and you want to perform a specific operation on each of them, depending on its type.

What is visitor pattern C++?

Visitor in C++ Visitor is a behavioral design pattern that allows adding new behaviors to existing class hierarchy without altering any existing code.

What is the difference between strategy and decorator pattern?

The strategy pattern allows you to change the implementation of something used at runtime. The decorator pattern allows you augment (or add to) existing functionality with additional functionality at run time.


1 Answers

In short

they come from to different conceptualizations that, in some languages where double dispatch is not natively supported, lead to the visitor pattern as a way to concatenate two (or more) single dispatch in order to have a multi-dispatch surrogate.

In long

The idea of multiple dispatch is - essentially - allow a call like

void fn(virtual base_a*, virtual base_b*); (note: not as a class member: this is NOT C++! )

that can be overridden as

void fn(virtual derived_a1*, virtual derived_b1*); void fn(virtual derived_a2*, virtual derived_b1*); void fn(virtual derived_a1*, virtual derived_b2*); void fn(virtual derived_a2*, virtual derived_b2*); 

so that, when calling

fn(pa, pb) 

the call is redirected to the override that matches the actual runtime type of both pa and pb. (You can generalize this to whatever number of parameters)

In language like C++, C#, Java, this mechanism does not exist and runtime type dispatching basically works with just one parameter (that, being just one, is made implicit in the function by making the function itself member of the class:

in other words, the pseudocode

void fn(virtual base_a*, base_b*)  

becomes the (real C++)

class base_a { public:     virtual void fn(base_b*); } 

Note that here there is no more virtual in front of base_b, that from now is static. A call like

pa->fn(pb) if pa points to a derived_a2 and pb to a derived_b1 will be dispatched to derived_a2::fn(base_b*), no matter if there is a derived_a2::fn(derived_b1*) in there: the run-time type of the object pointed by pb is not taken into account.

The idea of the visitor patter is that you call the virtual dispatch of an object that calls (eventually back) the virtual dispatch of another:

class base_a { public:    virtual void fn(base_b*)=0;    virtual void on_visit(derived_b1*)=0;    virtual void on_visit(derived_b2*)=0; };  class base_b { public:    virtual void on_call(derived_a1*)=0;    virtual void on_call(derived_a2*)=0; };  //forward declarations, to allow pointers free use in other decls. class derived_a1; class derived_b1;   class derived_a1: public base_a { public:    virtual void fn(base_b* pb) { pb->on_call(this); }    virtual void on_visit(derived_b1* p1) { /* useful stuff */ }    ... };  class derived_b1: public base_b { public:   virtual void on_call(derived_a1* pa1) { pa1->on_visit(this); }   ...  }; 

now, a call like pa->fn(pb), if pa points to derived_a1 and pb to derived_b1, will finally go to derived_a1::on_visit(derived_b1*).

like image 104
Emilio Garavaglia Avatar answered Nov 08 '22 15:11

Emilio Garavaglia