从 Objective-C 函数调用 C++ 函数不起作用
Calling C++ function from Objective-C function doesn't work
我不明白为什么这个从 Objective-C 文件中调用的简单 C++ 函数不起作用...如何解决这个问题?
上下文-menu.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];
}
app.h:
#ifdef __cplusplus
extern "C" {
#endif
void showMyCppMenu();
#ifdef __cplusplus
}
#endif
app.cpp:
#include "app.h"
void showMyMenu();
void showMyCppMenu() {
showMyMenu();
}
main.m:
#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
,而不是 _showMyMenu
就像在 C
.
中一样
要解决您的问题,您只需将函数标记为 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 中的定义包装起来,它会工作得很好。
我不明白为什么这个从 Objective-C 文件中调用的简单 C++ 函数不起作用...如何解决这个问题?
上下文-menu.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];
}
app.h:
#ifdef __cplusplus
extern "C" {
#endif
void showMyCppMenu();
#ifdef __cplusplus
}
#endif
app.cpp:
#include "app.h"
void showMyMenu();
void showMyCppMenu() {
showMyMenu();
}
main.m:
#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
,而不是 _showMyMenu
就像在 C
.
要解决您的问题,您只需将函数标记为 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 中的定义包装起来,它会工作得很好。