4

我对 Objective-C 相当陌生,我正在为我们的项目构建一个 iCloud 实用程序库。我在 Objective-C 中创建了一个 iCloud 实用程序类。我注意到有很多关于在 Objective-C 中使用 C++ 类的问题,但反之则很少。问题是,我们的项目(以及构建我们项目的引擎)完全是用 C++ 编写的,因此,我编写的 Objective-c icloud 类需要可以从我正在编写的 C++ 接口访问。

这是一个例子:

iCloudUtils.h

#import <Foundation/Foundation.h>

@interface iCloudUtil : NSObject
@property(nonatomic, assign, getter=isAvailable) BOOL iCloudIsAvailable;

@property (assign) NSString *urlToFile;

-(bool) init_iCloud;
-(void) loadDocument: (NSString*)fileUrl;
@end

iCloudUtils.m

#import "iCloudUtil.h"
@synthesize urlToFile = _urlToFile;
@synthesize iCloudIsAvailable = _iCloudIsAvailable;
//////////////////////////////////////////////////////


@implementation iCloudUtil
-(bool) init_iCloud{
//blah blah
}

-(void) loadDocument: (NSString*)fileUrl{
//more blah blah
}

@end

我试图从中使用它的 C++ 类是这样的:

iCloud接口.h

#ifndef __ICloudInterface__
#define __ICloudInterface__

class ICloudInterface {

    static ICloudInterface *_instance;
    bool iCloudIsAvailable;

public:
    ICloudInterface();
    ~ICloudInterface();

    bool init_iCloud();
};

#endif 

我想要做的是能够在我的 C++ 类中创建一个 iCloudUtils 的实例,但是我不能在我的 C++ 类中#include 它而不收到关于“Stray '@' in program”的 200 多个错误。我对 Objective-C 非常缺乏经验,更不用说将它与 C++ 混合,所以有人可以帮助我就如何实现这一目标提供建议吗?

编辑:只是为了清除它,我已经尝试将 .cpp 重命名为 .mm 但这并没有解决问题。

Edit2:另外,根据我对阅读的理解(如果我错了,请纠正我)将文件更改为.mm,任何包含它的文件也必须更改为.mm。我不能这样做,因为它需要更改非常大量的文件以适应单个可选类(提醒这个项目及其引擎都是 c++,这是一个跨平台的引擎和项目)。

4

4 回答 4

2

将您的 .cpp 文件名切换为 .mm - 这会将其从 c++ 转换为 Objective-c++。然后,您将编译一种能够理解标头的语言。

于 2013-02-21T17:59:21.920 回答
2

您不能在文件扩展名为 .cpp 的 C++ 文件中使用 Obj.C 类对象

您可以在 Obj.C 类中使用扩展名为 .mm 的 c++ 类

要在 C++ 和 Obj.C 类之间进行通信,请在 .mm 文件中编写 helper c++ 类

说:

  interface iCloudUtil : NSObject  // in .m file
  class ICloudInterface            //in cpp file.

您可以使用桥接类帮助在 .cpp 文件中的 cpp 类与 .m 文件中的 obj.C 类之间进行通信。

//include iCloudUtil.h only in ICloudBridge.mm file.

class ICloudBridge {

    static ICloudBridge *_instance;

public:
    ICloudBridge();
    ~ ICloudBridge();

    ICloudBridge *sharedICloudBridge();
    void loadDocument();
};

void  ICloudBridge:: loadDocument()
{
   iCloudUtil *obj = [iCloudUtil alloc] init] autorelese];
   [obj  loadDocument:@"DocName"];
}

在 iCloudInterface.cpp

ICloudBridge *bridge = ICloudBridge::sharedICloudBridge();
bridge-> loadDocument();
于 2013-02-21T18:19:54.770 回答
1

我认为您必须在源代码文件名中使用.mm扩展名而不是扩展名,或者将文件设置为在 Xcode 中编译为 Objective-C++。.m

于 2013-02-21T17:59:30.657 回答
0

当然——你可以做到这一点。需要的主要是将 .CPP 文件设置为“Objective-C++ Source”——“.mm”约定是一种自动的精巧,但不是必需的。然后,您可以随意混合和匹配:

#import <Foundation/Foundation.h>
#include <iostream>

@interface iCloudUtil : NSObject
@property(nonatomic, assign, getter=isAvailable) BOOL iCloudIsAvailable;
@property(assign) NSString *urlToFile;
- (bool) init_iCloud;
- (void) loadDocument:(NSString*)fileUrl;
@end

//////////////////////////////////////////////////////
@implementation iCloudUtil

@synthesize urlToFile = _urlToFile;
@synthesize iCloudIsAvailable = _iCloudIsAvailable;

- (bool) init_iCloud {
    NSLog(@"init_iCloud");
    std::cout << "i can mix std c++ in here as well\n";
    return YES;
}

- (void) loadDocument:(NSString*)fileUrl {
    NSLog(@"url is %@",fileUrl);
}

@end
//////////////////////////////////////////////////////

class ICloudInterface {
    static ICloudInterface* _instance;
    bool iCloudIsAvailable;
    iCloudUtil* myCloudUtil;
public:
    ICloudInterface();
    ~ICloudInterface();

    bool init_iCloud();
};

ICloudInterface::ICloudInterface() {
    std::cout << "ICloudInterface::ctor\n";
    myCloudUtil = [[iCloudUtil alloc] init];
    [myCloudUtil init_iCloud];
    [myCloudUtil loadDocument:@"file://foo/bar/baz"];
}

ICloudInterface::~ICloudInterface() {
    [myCloudUtil release], myCloudUtil = nil;
    std::cout << "ICloudInterface::dtor\n";
}

int main(int argc, const char * argv[]) {
    std::cout << "Test objc/objc++\n";
    ICloudInterface iface;
    return 0;
}
于 2013-02-21T19:23:06.137 回答