gpt4 book ai didi

ios - 何时需要将应用程序源包含在测试目标中?

转载 作者:IT老高 更新时间:2023-10-28 11:26:45 25 4
gpt4 key购买 nike

在一个新项目中我有这个简单的测试

#import <XCTest/XCTest.h>
#import "ViewController.h"

@interface ViewControllerTests : XCTestCase
@end

@implementation ViewControllerTests

- (void)testExample
{
// Using a class that is not in the test target.
ViewController * viewController = [[ViewController alloc] init];
XCTAssertNotNil(viewController, @"");
}

@end

ViewController.h 是 not 测试目标的一部分,但它编译并运行测试没有任何问题。

enter image description here

我认为这是因为首先构建应用程序(作为依赖项)然后是测试。然后链接器找出 ViewController 类是什么。

但是,在一个较旧的项目中,具有完全相同的测试和 ViewController 文件,构建在链接器阶段失败:

Undefined symbols for architecture i386:
"_OBJC_CLASS_$_ViewController", referenced from:
objc-class-ref in ViewControllerTests.o

即使创建了新的 XCTest 单元测试目标,也会发生此链接器错误。

为了解决这个问题,可以在应用程序和测试目标中都包含源(勾选上图中的两个框)。这会导致在模拟器的系统日志中出现重复符号的构建警告(打开模拟器并按 cmd-/来查看):

Class ViewController is implemented in both 
[...]/iPhone Simulator/ [...] /MyApp.app/MyApp and
[...]/Debug-iphonesimulator/LogicTests.octest/LogicTests.
One of the two will be used. Which one is undefined.

这些警告有时会导致以下示例所示的问题:

 [viewController isKindOfClass:[ViewController class]]; // = NO
// Memory address of the `Class` objects are different.

NSString * instanceClassString = NSStringFromClass([viewController class]);
NSString * classString = NSStringFromClass([ViewController class]);

[instanceClassString isEqualToString:classString]; // = YES
// The actual class names are identical

那么问题是旧项目中的哪些设置要求将应用程序源文件包含在测试目标中?


评论摘要

工作项目和非工作项目之间:

  1. 链接器输出没有区别(以Ld开头的命令)。
  2. 目标依赖没有区别(测试目标有1个依赖,也就是app)
  3. 链接器设置没有区别。

最佳答案

我花了一些时间解决这个问题。

如果您阅读 this documentation你会发现 Xcode 有两种运行测试的模式。逻辑测试和应用测试。不同之处在于逻辑测试使用您内置的类和符号构建自己的目标。生成的可执行文件可以在模拟器中运行并将测试输出报告回 Xcode。另一方面,应用程序测试构建了一个动态库,链接到您的代码,该代码在运行时注入(inject)到应用程序中。这允许您在 iPhone 环境中运行测试并测试 Xib 加载和其他内容。

因为当您取消链接源文件时测试目标中缺少符号,您的旧项目似乎有一个为逻辑测试配置的测试目标,而不是应用程序(单元)测试。

现在 Xcode 似乎是 trying not to distinguish between the two并默认创建一个应用程序测试目标,让我们了解您可能需要更改的所有内容,以将您的逻辑测试目标转变为单元测试目标。

我还将假设您有一个应用程序目标而不是静态库目标,因为方向会有所不同。

  1. 在测试目标的build设置中,删除“Bundle Loader”和“Test Host”build设置。稍后我们将让 Xcode 添加这些内容
  2. 您需要从测试目标中删除应用程序中的所有 .m 文件。您可以通过选择所有 .m 文件并在 Xcode 文件检查器中删除测试目标来执行此操作,也可以使用测试目标的编译源构建阶段。
  3. 为您的测试目标更改“框架搜索路径”。对于 Xcode 5,它们应该是$(SDKROOT)/Developer/Library/Frameworks
    $(继承)
    $(DEVELOPER_FRAMEWORKS_DIR)
    按这个顺序和with no extra quotes or backslashes
  4. 转到您的测试目标的build设置的“常规” Pane ,然后从下拉菜单中选择您的目标。如果菜单已经指定了您的应用程序目标,您应该将其关闭然后再打开。这将使 Xcode 使用正确的值重新配置 Bundle loader 和 Test Host 设置。
  5. 最后仔细检查您的应用程序的方案。在方案下拉列表中选择编辑方案。然后单击测试操作。确保您的测试目标位于信息 Pane 的列表中,并确保选择了所有测试。

此信息或多或少来自上述链接文档,但我更新了 Xcode 5 的步骤。

编辑:

嗯 100% 请注意 eph515 关于调试符号可见的说法,但您可能还想检查是否有人没有将您的方案的测试操作设置为在 Release 或其他配置中构建。单击方案选择器并选择编辑方案。单击测试操作,然后确保构建配置为 Debug

build configuration screen for test action in a scheme

如果您有静态库目标

所以如果你有一个静态库目标,你有两个选择:1. 逻辑测试2. 宿主应用中的应用测试

对于 1. 你必须确保 Bundle LoaderTest Host 对于你的静态库目标是空的。然后必须将您的源代码编译到测试目标中,因为它们没有其他方式可以运行。

对于 2. 您需要在 Xcode 中创建一个新的应用程序项目并将您的静态库项目添加为子项目。然后,您需要手动将 Bundle LoaderTest Host build设置从 New App 的测试目标复制到 Static Lib 测试目标。然后打开新测试应用程序的方案并将测试目标添加到新应用程序的测试操作中。要在您的 lib 上运行测试,请为您的主机应用程序运行测试操作。

关于ios - 何时需要将应用程序源包含在测试目标中?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21911168/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com