3

我有一个结构,里面有一个指向同一结构的函数的指针。现在我需要调用一个指向结构外函数的指针。我举一个下面的代码示例:

#include <iostream>

struct test {
    void (test::*tp)(); // I need to call this pointer-to-function
    void t() {
        std::cout << "test\n";
    }
    void init() {
        tp = &test::t;
    }
    void print() {
        (this->*tp)();
    }
};
void (test::*tp)();

int main() {
    test t;
    t.init();
    t.print();
    (t.*tp)(); // segfault, I need to call it
    return 0;
}
4

2 回答 2

7

(t.*tp)();正在尝试调用tp在全局命名空间 as 中定义的成员函数指针void (test::*tp)();,请注意它实际上已初始化为空指针(通过零初始化1),调用它会导致UB,一切皆有可能。

如果要在对象上调用(即)的数据成员tp,则应将其更改为tt.tpt

(t.*(t.tp))();
     ^
     |
     ---- object on which the member function pointed by tp is called

如果确实要调用 global tp,则应适当地对其进行初始化,例如

void (test::*tp)() = &test::t;

那么你就可以

(t.*tp)(); // invoke global tp on the object t

1关于零初始化

在以下情况下执行零初始化:

1) 对于每个具有静态或线程本地存储持续时间的命名变量that is not subject to constant initialization (since C++14),在任何其他初始化之前。

于 2018-10-14T12:44:46.040 回答
0

@songyuanyao 的回答是有效的。但是,您确定要以这种方式使用您的结构吗?为什么不直接使用继承和虚方法呢?:

class base_test {
public:
    virtual void t() { std::cout << "test\n"; }
    void print() { t(); }
};

然后你可以子类化它:

class my_test : base_test {
public:
    virtual void t() { std::cout << "my test\n"; }
};

在您的main()函数(或任何地方)中,您可以有函数返回指向基类的指针或引用,它们实际上是子类的实例。这样,您就不必担心指针。

缺点是您必须在编译时了解您的不同测试(甚至在使用站点,正如我刚刚解释的那样)。如果你这样做,我会用常见的成语。

于 2018-10-14T12:50:47.613 回答