2016-07-09 8 views
2

Objective-Cファイルからこの単純な呼び出し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      [email protected] 

./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 

最初のC++関数が別のobjective-c関数を呼び出している間、main.m objective-cからC++関数を呼び出す必要があります。

ビルドスクリプトの修正方法は?それはなぜ機能しないのですか?

更新: 問題を解決した後に見つけました。 CとC++を混合するための良い材料:そのコンパイラはC++シンボルとしてシンボルをエクスポートして、あなたのapp.cppから http://yosefk.com/c++fqa/mixing.html

+0

showMyCppMenuを含むファイルをリンクするのを忘れましたか? –

+1

Cシンボルは '畳み込まれていません'しかし、C++シンボルは '畳み込まれています'シンボルの '畳み込み'のプロセスは、シンボルタイプなどの情報を前に付けることです。 – user3629249

答えて

4

showMyMenuは、C関数としてマークされていない:showMyMenu、それはCになるようではない_showMyMenu

はあなたが必要とするあなたの問題を解決するために、単にC関数として機能マーク:

app.cppで
// app.cpp 
#include "app.h" 

extern "C" void showMyMenu(); 

void showMyCppMenu() { 
    showMyMenu(); 
} 
+0

また、 'showMyMenu ) 'を' app.cpp'内に定義するだけでなく 'context-menu.mm'の' showMyMenu() 'の定義を' extern "C" void showMyMenu(){...} '別の方法で' context -menu.mm'〜 'context-menu.m' – Zelid

2

、あなたがC++アプリケーションのコンテキストで「showMyMenu」を宣言しているが、それがリンクされていますC++の名前が変更された '__Z10showMyMenu'関数ではなく、あなたが意図している 'C'関数として扱われます。

あなたが含まれていmain.h(または類似のもの)を定義した場合:

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

または単に同じスコープルールとapp.cに定義をラップし、それが正常に動作します。

関連する問題