Functions Pointers by example (en)
Today, a small memento on the different ways to practice Function Pointers in C++.
A function pointer is simply the memory address where a function is located. If our program calls code at this address, it will execute the related function.
And as examples are better than a long explanation, let’s see some examples :
Pointers to simple functions :
We start by creating two functions for basic mathematical operations (addition, multiplication).
These two operations have the same profile (parameters, return value).
int multiply (int value1, value2 int)
{
return value1 * value2;
}
int add (int value1, value2 int)
{
value1 + value2 return;
}
The goal now is to delegate the processing of this transaction to allow a call like this :
function (number_1, number_2, type_of_operation)
So, let’s create this function :
void calculate (int value1, value2 int, int (* operation) (int, int))
{
int res = operation (value1, value2);
cout << endl << res;
}
Some explanations :
- The two first parameters are integers (value1 and value2).
- Third parameter is a function pointer :
- it takes two integers as parameters.
- it returns an integer.
- The calculated function call the third parameter function with the 2 first parameters.
We can now call the calculation function :
int main ()
{
calculation (2,3, multiply);
calculation (2.3, add);
cout << "EXIT SUCCESS" << endl;
return 0;
}
Tip: With a typedef, you can make the writing of this code a bit more readable:
typedef int (* Operation) (int, int);
/* This allows to write the following prototype : */
void calculate (int value1, value2 int, Operation operation);
/* Instead of : */
void calculate (int value1, value2 int, int (* operation) (int, int))
Pointers to functions (but not only):
With templates :
The objective here is to do the work at compile rather than during the execution.
The calculation function is going to take a template corresponding to an operation:
template <typename operation = "">
void calculate (int value1, value2 int, OPERATION operation)
{
int res = operation (value1, value2);
cout << endl << res;
}
The advantage is that you can now pass objects such as the following :
struct multiplication
{
int operator () (int value1, value2 int)
{
return value1 * value2;
}
};
int main ()
{
multiplication multiply;
computation (2,3 addition);
calculation (2,3, multiply);
cout << "EXIT SUCCESS" << endl;
return 0;
}
With std :: function (from c ++ 0x) :
Another way is to use the function type of the stl (#include). The calculation function becomes:
void calculate (int value1, value2 int, std :: function <int> operation)
// We pass in a parameter function which is called operation that returns an integer and takes two integer parameter.
{
int res = operation (value1, value2);
cout << endl << res;
}
The corresponding call is exactly the same as before :
int main ()
{
multiplication multiply;
calculation (3.2, addition);
calculation (3.2, multiply);
cout << "EXIT SUCCESS" << endl;
return 0;
}
But the novelty is that the lamda can also be used (or anonymous functions). For example:
int main ()
{
std :: function func = [] (int value1, value2 int) {return value1 * value2;}; // [&] If you want to return the result by reference.
calculation (3.2, func);
// Faster:
calculation (3.2, [] (int value1, value2 int) {return value1 value2-;});
cout << "EXIT SUCCESS" << endl;
return 0;
}
And, it’s over.
This article may contain mistakes, do not hesitate to let me know.