web-dev-qa-db-fra.com

Passer un pointeur à une fonction de membre de classe comme paramètre

J'ai écrit un petit programme où j'essaie de transmettre un pointeur à la fonction membre d'une classe à une autre fonction. Pouvez-vous m'aider s'il vous plaît et où je vais mal ..?

#include<iostream>
using namespace std;
class test{
public:
        typedef void (*callback_func_ptr)();
        callback_func_ptr cb_func;

        void get_pc();

        void set_cb_ptr(void * ptr);

        void call_cb_func();
};
void test::get_pc(){
         cout << "PC" << endl;
}
void test::set_cb_ptr( void *ptr){
        cb_func = (test::callback_func_ptr)ptr;
}
void test::call_cb_func(){
           cb_func();
}
int main(){
        test t1;
            t1.set_cb_ptr((void *)(&t1.get_pc));
        return 0;
}

Je reçois l'erreur suivante lorsque j'essaie de le compiler.

error C2276: '&' : illegal operation on bound member function expression
15
ajay bidari

Vous ne pouvez pas lancer un pointeur de fonction sur void*.

Si vous souhaitez qu'un pointeur de fonction pointe sur une fonction membre, vous devez déclarer le type comme

ReturnType (ClassType::*)(ParameterTypes...)

En outre, vous ne pouvez pas déclarer un pointeur de fonction à une fonction membre liée, par ex.

func_ptr p = &t1.get_pc // Error

Au lieu de cela, vous devez obtenir l'adresse comme celle-ci:

func_ptr p = &test::get_pc // Ok, using class scope.

Enfin, lorsque vous appelez à un pointeur de fonction pointant vers une fonction membre, vous devez l'appeler avec une instance de la classe que la fonction est membre de. Par exemple:

(this->*cb_func)(); // Call function via pointer to current instance.

Voici l'exemple complet avec toutes les modifications appliquées:

#include <iostream>

class test {
public:
    typedef void (test::*callback_func_ptr)();
    callback_func_ptr cb_func;
    void get_pc();
    void set_cb_ptr(callback_func_ptr ptr);
    void call_cb_func();
};

void test::get_pc() {
    std::cout << "PC" << std::endl;
}

void test::set_cb_ptr(callback_func_ptr ptr) {
    cb_func = ptr;
}

void test::call_cb_func() {
    (this->*cb_func)();
}

int main() {
    test t1;
    t1.set_cb_ptr(&test::get_pc);
    t1.call_cb_func();
}
19
Snps

En plus de la réponse de SNPS, vous pouvez également utiliser un wrapper de fonction de C++ 11 pour stocker une fonction Lambda:

#include <iostream>
#include <functional>

class test
{
  public:
   std::function<void ()> Func;
   void get_pc();
   void call_cb_func();
   void set_func(std::function<void ()> func);
};

void test::get_pc()
{
  std::cout << "PC" << std::endl;
}

void test::call_cb_func()
{
  Func();
}

void test::set_func(std::function<void ()> func)
{
  Func = func;
}

int main() {
  test t1;
  t1.set_func([&](){ t1.get_pc(); });
  t1.call_cb_func();
}
2
Attaque