Указатель функции C ++ (член класса) на нестатическую функцию-член - PullRequest
39 голосов
/ 13 июня 2009
class Foo {
public:
    Foo() { do_something = &Foo::func_x; }

    int (Foo::*do_something)(int);   // function pointer to class member function

    void setFunc(bool e) { do_something = e ? &Foo::func_x : &Foo::func_y; }

private:
    int func_x(int m) { return m *= 5; }
    int func_y(int n) { return n *= 6; }
};

int
main()
{
    Foo f;
    f.setFunc(false);
    return (f.*do_something)(5);  // <- Not ok. Compile error.
}

Как мне заставить это работать?

Ответы [ 6 ]

35 голосов
/ 13 июня 2009
 class A{
    public:
        typedef int (A::*method)();

        method p;
        A(){
            p = &A::foo;
            (this->*p)(); // <- trick 1, inner call
        }

        int foo(){
            printf("foo\n");
            return 0;
        }
    };

    void main()
    {
        A a;
        (a.*a.p)(); // <- trick 2, outer call
    }
32 голосов
/ 13 июня 2009

Строка, которую вы хотите, это

   return (f.*f.do_something)(5);

(Это компилируется - я пробовал)

"*f.do_something" относится к самому указателю --- "f" указывает нам, где получить значение do_something из Но нам все еще нужно дать объект, который будет указателем this при вызове функции. Вот почему нам нужен префикс "f.".

3 голосов
/ 06 ноября 2012
class A {
    int var;
    int var2;
public:
    void setVar(int v);
    int getVar();
    void setVar2(int v);
    int getVar2();
    typedef int (A::*_fVar)();
    _fVar fvar;
    void setFvar(_fVar afvar) { fvar = afvar; }
    void insideCall() { (this->*fvar)(); }
};

void A::setVar(int v)
{
    var = v;
}

int A::getVar()
{
    std::cout << "A::getVar() is called. var = " << var << std::endl;
    return var;
}

void A::setVar2(int v2)
{
    var2 = v2;
}

int A::getVar2()
{
    std::cout << "A::getVar2() is called. var2 = " << var2 << std::endl;
    return var2;
}

int main()
{
    A a;
    a.setVar(3);
    a.setVar2(5);

//    a.fvar = &A::getVar;
    a.setFvar(&A::getVar);
    (a.*a.fvar)();

    a.setFvar(&A::getVar2);
    (a.*a.fvar)();

    a.setFvar(&A::getVar);
    a.insideCall();

    a.setFvar(&A::getVar2);
    a.insideCall();

    return 0;
}

Я расширил ответ Ника Дандулакиса. Спасибо.

Я добавил функцию, которая устанавливает указатель на функцию-член извне класса. Я добавил еще одну функцию, которую можно вызывать извне, чтобы показать внутренний вызов указателя на функцию-член.

0 голосов
/ 11 мая 2016
#include<iostream>
using namespace std;

class A {

public:
    void hello()
    {
        cout << "hello" << endl;
    };

    int x = 0;

};


void main(void)
{

    //pointer
    A * a = new A;
    void(A::*pfun)() = &A::hello;
    int  A::*v1 = &A::x;

    (a->*pfun)();
    a->*v1 = 100;
    cout << a->*v1 << endl << endl;

    //----------------------------- 
    A  b;
    void(A::*fun)() = &A::hello;
    int  A::*v2 = &A::x;

    (b.*fun)();
    b.*v2 = 200;
    cout << b.*v2 << endl;

}
0 голосов
/ 22 октября 2009

Я думаю, что вызов нестатического члена класса также может быть выполнен с использованием статической функции-члена.

0 голосов
/ 13 июня 2009

Try (f. * Do_something) (5);

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...