从 Objective-C 函数调用C++函数不起作用

Calling C++ function from Objective-C function doesn't work

本文关键字:函数 不起作用 C++ 函数调用 Objective-C      更新时间:2023-10-16

我不知道为什么这个简单的从 Objective-C 文件调用C++函数不起作用......如何解决问题?

上下文菜单.m:

#import <Cocoa/Cocoa.h>
void showMyMenu() {
    NSMenu *theMenu = [[NSMenu alloc] initWithTitle:@"Contextual Menu"];
    [theMenu insertItemWithTitle:@"Beep" action:@selector(beep:) keyEquivalent:@"" atIndex:0];
    [theMenu insertItemWithTitle:@"Honk" action:@selector(honk:) keyEquivalent:@"" atIndex:1];
    [theMenu popUpMenuPositioningItem:nil atLocation:[NSEvent mouseLocation] inView:nil];
}

应用h:

#ifdef __cplusplus
extern "C" {
#endif
    void showMyCppMenu();
#ifdef __cplusplus
}
#endif

应用.cpp:

#include "app.h"
void showMyMenu();
void showMyCppMenu() {
    showMyMenu();
}

主:

#import <Cocoa/Cocoa.h>
#include "app.h"
// void showMyMenu();
// void showMyCppMenu();
int main(int argc, const char * argv[])
{
    NSApplication * application = [NSApplication sharedApplication];
    // NSView* ns = (NSView*) startup();
    [application setActivationPolicy:NSApplicationActivationPolicyRegular];
    id ns = [[NSView new] autorelease];
    id menubar = [[NSMenu new] autorelease];
    id appMenuItem = [[NSMenuItem new] autorelease];
    [menubar addItem:appMenuItem];
    [application setMainMenu:menubar];
    id appMenu = [[NSMenu new] autorelease];
    id appName = [[NSProcessInfo processInfo] processName];
    id quitTitle = [@"Quit " stringByAppendingString:appName];
    id quitMenuItem = [[[NSMenuItem alloc] initWithTitle:quitTitle
        action:@selector(terminate:) keyEquivalent:@"q"] autorelease];
    [appMenu addItem:quitMenuItem];
    [appMenuItem setSubmenu:appMenu];
    [[ns window] setTitle:appName];
    [[ns window] makeKeyAndOrderFront:nil];
    [NSApp activateIgnoringOtherApps:YES];
    [NSApp activateIgnoringOtherApps:YES];
    NSStatusItem * statusItem = [[[NSStatusBar systemStatusBar] statusItemWithLength:NSSquareStatusItemLength] retain];
    [statusItem setMenu:appMenu];
    // [statusItem setImage:icon];
    // [statusItem setAlternateImage:icon2];
    [statusItem setHighlightMode:YES];
    // [statusItem setToolTip:[NSString stringWithUTF8String:title]];
    // showMyMenu();
    showMyCppMenu();
    [application run];
    return EXIT_SUCCESS;
}

build.sh:

gcc -fPIC context-menu.m app.cpp -framework Cocoa -x objective-c -c -lobjc -lstdc++ 
ar rcs libapp.a context-menu.o app.o
gcc -L/Users/alex/Workspace/SimpleAppFromScratch/mixing-objc1 main.m -framework Cocoa -x objective-c -o main -lobjc -lstdc++ -lapp

当我运行 ./build.sh 时,我总是收到一个错误:

    Undefined symbols for architecture x86_64:
      "showMyMenu()", referenced from:
          _showMyCppMenu in libapp.a(app.o)
    ld: symbol(s) not found for architecture x86_64

但是_showMyCppMenu符号在libapp.a里面:

nm ./libapp.a                      alex@Pangaea
./libapp.a(context-menu.o):
                 U _OBJC_CLASS_$_NSEvent
                 U _OBJC_CLASS_$_NSMenu
                 U ___CFConstantStringClassReference
                 U _objc_msgSend
0000000000000000 T _showMyMenu
./libapp.a(app.o):
                 U __Z10showMyMenuv
0000000000000000 T _showMyCppMenu

我真的需要从main.m objective-c调用c ++函数,而第一个c ++函数正在调用另一个objective-c函数。

如何修复构建脚本?为什么不起作用?

更新:解决问题后找到了。C++与C混合的良好材料:http://yosefk.com/c++fqa/mixing.html

app.cpp中的showMyMenu不会标记为C函数,因此编译器将符号导出为C++符号:showMyMenu,而不是像C那样_showMyMenu

要解决您的问题,您只需将函数标记为C函数:

// app.cpp
#include "app.h"
extern "C" void showMyMenu();
void showMyCppMenu() {
    showMyMenu();
}

在 app.cpp 中,您在C++应用程序的上下文中声明"showMyMenu",因此它被链接到为C++名称损坏的"__Z10showMyMenu"函数,而不是您想要的"C"函数。

如果定义包含以下内容的 main.h(或类似内容):

extern "C" {
   void showMyMenu(void);
};

或者只是使用相同的范围规则将定义包装在 app.c 中,它将正常工作。