Click here to Skip to main content
Licence CPOL
First Posted 26 Nov 2010
Views 11,379
Downloads 141
Bookmarked 17 times

C++0x Dynamic Message Passing

By | 26 Nov 2010 | Article
C++0x Dynamic Message Passing Ala Objective-C

Introduction

Sometimes it is useful to have dynamic message passing as in Objective-C. The small header presented below allows any class to contain a dynamic message map that can be used to add methods dynamically to an object.

Background

Objective-C is deemed more flexible than C++ because it allows a form of dynamic dispatch known as message passing.

Message passing is a form of dynamic dispatch that does not require implementation of a specific interface type. When sending a message to a target object, it is unknown to the compiler if the message can be handled by the object or not. It is only during the execution of the program that conformance to an interface is discovered.

Message passing is quite flexible because it does not require a lot of planning. Objective-C has been praised for this feature as more flexible than C++. This article demonstrates how easy it is to do the same in standard C++ (c++0x).

Using the Code

Using the code is very easy. The following steps have to be followed:

  • Include the header "mp_object.hpp" in your project.
  • Inherit from class mp_object.
  • Add dynamic methods to your object by using the method 'add_method'.

In order to add methods to an object, you need a prototype function. A prototype function can be any free-standing function. The following code is an example of how to declare prototypes and add dynamic methods to your code:

#include <iostream>
#include "mp_object.hpp"
using namespace std;

//prototype method
void draw() {}

//prototype method
void setColor(int color) {}

//rectangle
class rect : public mp_object {
public:
    //constructor
    rect() {
        add_method(::draw, &rect::draw);
        add_method(::setColor, &rect::setColor);
    }

    //draw
    void draw() {
        cout << "rect\n";
    }

    //set color
    void setColor(int color) {
        cout << "rect color = " << color << "\n";
    }
};

//circle
class circle : public mp_object {
public:
    //constructor
    circle() {
        add_method(::draw, &circle::draw);
        add_method(::setColor, &circle::setColor);
    }

    //draw
    void draw() {
        cout << "circle\n";
    }

    //set color
    void setColor(int color) {
        cout << "circle color = " << color << "\n";
    }
};

You can invoke an object dynamically like this:

int main() {
    rect r;
    circle c;

    r.invoke(draw);
    c.invoke(draw);

    r.invoke(setColor, 10);
    c.invoke(setColor, 20);

    return 0;
}

The output of the above is:

rect
circle
rect color = 10
circle color = 20

Implementation Details

The implementation is very simple, really. The class mp_object has a hash map of prototype function pointers to method pointers. When a method is invoked, the appropriate method pointer is retrieved from the prototype function and executed.

In order to minimize memory usage, the class mp_object uses copy on-write: a method map is shared between many instances, until it is modified. If modified, the method map is copied and the change is applied to the copy.

The mp_object code is 39 lines of code, thanks to C++0x. If a previous version of C++ was used, then a lot of boilerplate code would be needed for doing essentially the same thing.

The code can easily be expanded with new capabilities; for example, querying an object if it supports a specific method or a form of reflection (combined with macros). These are beyond the scope of this short article, though.

History

  • 26th November, 2010: Initial post

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)

About the Author

Achilleas Margaritis

Software Developer (Senior)

Greece Greece

Member



Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
You must Sign In to use this message board. (secure sign-in)
 
Search this forum  
 FAQ
    Noise  Layout  Per page   
  Refresh
GeneralMy vote of 2 PinmemberWessam Fathi4:33 14 Dec '10  
GeneralGood idea, also I suggest putting the "message" functions in a separate namespace Pinmemberwtwhite1:49 9 Dec '10  
GeneralRe: Good idea, also I suggest putting the "message" functions in a separate namespace PinmemberAchilleas Margaritis2:10 9 Dec '10  
GeneralMy vote of 5 PinassociateJim Crafton9:22 2 Dec '10  
GeneralReally good start PinassociateJim Crafton9:16 2 Dec '10  
GeneralRe: Really good start PinmemberAchilleas Margaritis2:28 3 Dec '10  
GeneralMy vote of 2 PinmemberxComaWhitex9:57 29 Nov '10  
GeneralMy vote of 4 Pinmemberalex_dacosta2:20 29 Nov '10  
GeneralMy vote of 1 Pinmembergbb2110:36 27 Nov '10  

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.

Permalink | Advertise | Privacy | Mobile
Web01 | 2.5.120517.1 | Last Updated 26 Nov 2010
Article Copyright 2010 by Achilleas Margaritis
Everything else Copyright © CodeProject, 1999-2012
Terms of Use
Layout: fixed | fluid