Kev*_*ang 4 c++ xcode linker-errors ocmock
我正在使用OCMock 3.0.2,我通过cocoapods安装了我的测试目标:
platform :ios, '7.0'
xcodeproj 'myProject.xcodeproj'
target :myTestTarget do
pod 'OCMock', '~> 3.0.2'
end
link_with "myTestTarget"
Run Code Online (Sandbox Code Playgroud)
在我的测试文件(myTest.mm)中,我已经包含了OCMock并希望尝试新的就地验证策略,如下所示:
- (void) test_myTest
{
MyObject *obj = [MyObject new];
id robotMock = OCMPartialMock(obj);
[obj testMethod];
// some asserts
OCMVerify([obj _internalMethodToBeCalled]);
}
Run Code Online (Sandbox Code Playgroud)
到目前为止似乎正常.但是,当我尝试运行此特定测试用例时,我收到链接器错误:
Undefined symbols for architecture i386:
"OCMMakeLocation(objc_object*, char const*, int)", referenced from:
-[MyTests test_myTest] in MyTests.o
ld: symbol(s) not found for architecture i386
clang: error: linker command failed with exit code 1 (use -v to see invocation)
Run Code Online (Sandbox Code Playgroud)
我确认已正确引入OCMock并且也引用了OCMLocation.h/m文件.我看到OCMMakeLocation似乎是一个外部函数,但.m文件作为我的pods构建目标中的依赖项目存在,但不知何故它没有被链接.我必须让我的测试成为.mm,因为我包含了一些c ++文件.为什么这会成为OCMock的问题?
OCMMakeLocation是这样声明的
OCMLocation.h:
extern OCMLocation *OCMMakeLocation(id testCase, const char *file, int line);
Run Code Online (Sandbox Code Playgroud)
OCMLocation.m:
OCMLocation *OCMMakeLocation(id testCase, const char *fileCString, int line)
{
return [OCMLocation locationWithTestCase:testCase file:[NSString stringWithUTF8String:fileCString] line:line];
}
Run Code Online (Sandbox Code Playgroud)
它是在Objective-C接口之外定义的直接C函数.我不太了解编译器实际上在做什么(也许其他人可以更好地解释它),但据我所知,这就是正在发生的事情:你的测试文件是一个Objective-C++文件,所以它已经得到了符合C++链接,它确实名称为mangling(请参阅有关名称修改的信息).但是,OCMLocation被编译为Objective-C文件,因此它获得C链接,而不是C++链接,因此没有名称重整.因为你的测试符合C++链接,它会引入OCMock.h并假设它也是一个C++头,所以它假定编译它的源的结果将是相同的,它不会是.
简而言之,为了解决这个问题,您需要做的就是告诉编译器OCMock.h是测试文件中的C头:
#ifdef __cplusplus
extern "C" {
#endif
#import <OCMock/OCMock.h>
#ifdef __cplusplus
}
#endif
Run Code Online (Sandbox Code Playgroud)