【问题标题】:How to call a C++ function pointer from C?如何从 C 调用 C++ 函数指针?
【发布时间】:2010-06-21 09:56:52
【问题描述】:

我成功地终于能够在previous post here 的 C++ 项目中使用来自 PDcurses 库的 TUI 标头。

现在在我的班级中,我包含了 C 标头:

#include "tui.h"

tui 是 C 语言并且有这个菜单定义:

typedef struct 
{
    char *name; /* item label */
    FUNC  func; /* (pointer to) function */
    char *desc; /* function description */
} menu;

所以在 MainView.cpp 我有:

void sub0()
{
    //domenu(SubMenu0);
}

void sub1()
{
    //domenu(SubMenu1);
}


void MainView::showMainMenu()
{
    menu MainMenu[] =
    {
        { "Users", sub0, "Manage Users" },
        { "Accounts", sub1, "Manage Accounts" },
        { "Items", sub1, "Manage Items" },
        { "Staff", sub1, "Manage Staff" },
        { "", (FUNC)0, "" }  
    };
    startmenu(MainMenu, "Library System 1.0");
}

这是按预期工作的。

问题是我需要在 sub0() 和 sub1() 中调用我的类方法。

我尝试为我的班级定义 C++ 方法来尝试替换 sub0 和 sub1 与:

void MainView::sub0()
{
    //domenu(SubMenu0);
}
void MainView::sub1()
{
    //domenu(SubMenu1);
}

编译器抛出此错误:

error C2440: 'initializing' : cannot convert from 'overloaded-function' to 'FUNC'
None of the functions with this name in scope match the target type

将这些函数指针传递给 C 代码并消除该错误的最佳方法是什么?

谢谢

【问题讨论】:

  • 我不熟悉 TUI,但我不确定 boost::bind 是你想要的 - 在 MainMenu 初始化之后,绑定结构将被销毁并且你的 menu::func 实例将包含悬空指针。您可以创建一个 boost::function 对象数组,使用 boost:bind 填充新的函数数组,并使用 menu::func 的函数数组填充 MainMenu。对不起,这个评论应该是给 SoapBox 的

标签: c++ c


【解决方案1】:

C++ 类对象有一个“this”指针,它作为第一个参数不可见地传递给类成员函数。这意味着您不能将非静态成员函数用于接受 0 个参数的 C 样式函数指针。

所以解决这个问题,你有几个选择。在告诉你选择哪一个之前,我需要更多细节。

  • 将要在 C 代码中使用的函数设为静态。
  • 使用 boost 或 tr1 或其他允许将成员函数与 this 指针绑定的 C++ 库。使用 boost,它看起来像:

    { "Users", boost::bind(&MainView::sub0, this), "Manage Users" },
    
  • 您也许可以修改代码以传递对类对象的引用以及这些函数指针。如果是这样,您可以直接在您想要的对象上调用您的回调。不过,您似乎正在处理一个窗口管理器,因此您可能无法进行所需的更改以使这种方法正常工作。

  • 可能您最终要做的是某种解决方法,包括将指向您正在使用的对象的指针存储在某处,然后编写访问该指针的小存根函数,然后在其上调用成员函数。一些(我敢说,大多数)窗口管理器允许您为此目的在回调中包含指向任意数据的指针。

【讨论】:

  • boost::bind 不会产生函数指针——它会产生一个带有一些奇怪模板类型的函数 object。因此,它不能传递给 C。
  • 感谢 SoapBox。我选择了第一个选择,将功能设为静态,现在一切正常。
【解决方案2】:

我认为不可能将非静态成员函数作为 FUNC 指针传递。由于编译器会将 this 指针传递给所有非静态成员函数。

你有两个选择:

  1. 将您的 sub0 和 sub1 保留在课堂上并使其成为静态
  2. 将您的 sub0 和 sub1 移出课堂。如果您需要访问 sub0/sub1 中的其他成员变量/函数,则将指向您的类(又名 this)的指针传递给它们。

【讨论】:

  • 感谢 Jason,提出了很好的建议。正如您所说,将这些方法设为静态是解决方案。
猜你喜欢
  • 1970-01-01
  • 2011-01-29
  • 2012-04-16
  • 1970-01-01
  • 2022-12-05
  • 1970-01-01
  • 2016-01-01
  • 1970-01-01
  • 2018-04-16
相关资源
最近更新 更多