C ++从类外部设置指向结构中函数成员的指针

时间:2015-12-22 14:13:04

标签: c++ struct function-pointers member-function-pointers

我尝试通过方法SetPtr()从类外部设置str结构中包含的函数指针。

我收到错误:无效使用非静态成员函数。

class C {
    public:

    float v1, v2;

    struct S {
        float (*Read)();
        void  (*Write)(float);
    } str;

    float ReadV1() {
        return v1;
    }

    void WriteV1(float value) {
        v1 = value;
    }

    float ReadV2() {
        return v2;
    }

    void WriteV2(float value) {
        v2 = value;
    }

    void SetPtr(float (*read)(), void (*write)(float)) {
        str.Read = read;
        str.Write = write;
    }   

    void F()
    {
        float f = str.Read();
        str.Write(f);       
    }
};

int main()
{
    C c;

    c.SetPtr(c.ReadV1, c.WriteV2); // ERROR

    c.v1 = 0;
    c.F();
    return 0;
}

我还尝试通过指向成员函数的指针替换函数指针:

class C {
    public:

    float v1, v2;

    struct S {
        float (C::*Read)();
        void  (C::*Write)(float);
    } str;

    float ReadV1() {
        return v1;
    }

    void WriteV1(float value) {
        v1 = value;
    }

    float ReadV2() {
        return v2;
    }

    void WriteV2(float value) {
        v2 = value;
    }

    void SetPtr(float (C::*read)(), void (C::*write)(float)) {
        str.Read = read;
        str.Write = write;
    }   

    void F()
    {
        float f = str.Read();   // ERROR
        str.Write(f);           // ERROR
    }
};

int main()
{
    C c;

    c.SetPtr(&C::ReadV1, &C::WriteV2);

    c.v1 = 0;
    c.F();
    return 0;
}

但这会在课堂内移动错误:

  

错误:必须使用'。'或' - > '来调用指向成员的指针函数   '((C *)this) - > C :: str.C :: S :: Read(...)',例如'(... - > *   ((C *)this) - > C :: str.C :: S :: Read)(...)'

无论我尝试this->, braces, *, ->, .的任何组合,它都不起作用。

有什么想法吗? 谢谢!

1 个答案:

答案 0 :(得分:0)

您需要使用第二种形式(使用指向类方法的指针),但在调用您需要使用的方法时:

float f = (this->*str.Read)();
(this->*str.Write) (f);

第一种方法不能正常工作,因为指向类方法的指针不会衰减到指向标准函数的指针(即float (C::*) ()不能衰减到float (*) ())。

使用C++11,您可以将方法存储为std::function并使用std::bind

#include <functional>

class C {

    struct S {
        std::function <float()> Read ;
        std::function <void(float)> Write ;
    } str ;

    void SetPtr(float (C::*read)(), void (C::*write)(float)) {
        str.Read = std::bind(read, this);
        str.Write = std::bind(write, this, std::placeholders::_1);
    }   

    void F() {
        float f = str.Read();
        str.Write (f);
    }

}

int main () {
    C c;
    c.SetPtr(&C::ReadV1, &C::WriteV2);
}