ios - 在 Xcode 7 中更改了 +load 方法顺序

标签 ios objective-c unit-testing xcode7 objective-c-runtime

我发现 Xcode 7(版本 7.0 (7A220))更改了 +load 的顺序在单元测试期间调用类和类别的方法。

如果属于测试目标的类别实现了 +load方法,它现在在最后调用,此时类的实例可能已经被创建和使用。

我有一个 AppDelegate , 它实现了 +load方法。 AppDelegate.m文件还包含 AppDelegate (MainModule)类别。此外,还有一个单元测试文件 LoadMethodTestTests.m ,其中包含另一个类别 – AppDelegate (UnitTest) .

这两个类别还实现了 +load方法。第一类属于主要目标,第二类属于测试目标。

代码

我做了一个小test project来证明这个问题。 这是一个空的默认 Xcode 单 View 项目,仅更改了两个文件。

AppDelegate.m:

#import "AppDelegate.h"

@implementation AppDelegate

+(void)load {
    NSLog(@"Class load");
}

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    NSLog(@"didFinishLaunchingWithOptions");

    return YES;
}

@end

@interface AppDelegate (MainModule)
@end

@implementation AppDelegate (MainModule)

+(void)load {
    NSLog(@"Main Module +load");
}

@end

和一个单元测试文件(LoadMethodTestTests.m):

#import <UIKit/UIKit.h>
#import <XCTest/XCTest.h>
#import "AppDelegate.h"

@interface LoadMethodTestTests : XCTestCase

@end

@interface AppDelegate (UnitTest)
@end

@implementation AppDelegate (UnitTest)

+(void)load {
    NSLog(@"Unit Test +load");
}

@end

@implementation LoadMethodTestTests

-(void)testEmptyTest {
    XCTAssert(YES);
}

@end

测试

我在 Xcode 6/7 上对这个项目进行了单元测试(代码和 github 链接在下面)并得到以下 +load调用顺序:

Xcode 6 (iOS 8.4 simulator):
    Unit Test +load
    Class load
    Main Module +load
    didFinishLaunchingWithOptions

Xcode 7 (iOS 9 simulator):
    Class load
    Main Module +load
    didFinishLaunchingWithOptions
    Unit Test +load

Xcode 7 (iOS 8.4 simulator):
    Class load
    Main Module +load
    didFinishLaunchingWithOptions
    Unit Test +load

问题

Xcode 7运行测试目标类+load方法 ( Unit Test +load ) 最后,在 AppDelegate 之后已经创建。 这是正确的行为还是应该发送给 Apple 的错误?

可能没有指定,所以编译器/运行时可以自由地重新安排调用? 我看了看 this SO question以及 +load description in the NSObject documentation但我不太明白 +load当类别属于另一个目标时,方法应该起作用。

或者可能是AppDelegate出于某种原因是某种特殊情况吗?

为什么我要问这个

  1. 教育目的。
  2. 我曾经在单元测试目标内的类别中执行方法调配。现在,当调用顺序改变时,applicationDidFinishLaunchingWithOptions在混合发生之前执行。我相信还有其他方法可以做到这一点,但它在 Xcode 7 中的工作方式对我来说似乎违反直觉。我认为当一个类被加载到内存中时,+load这个类和+load在我们可以使用这个类之前,应该调用它所有类别的方法(比如创建一个实例并调用 didFinishLaunching... )。

最佳答案

TL,DR:这是 xctest 的错,而不是 objc 的错。

这是因为 xctest 可执行文件(实际运行单元测试的那个,位于 $XCODE_DIR/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/Library/Xcode/Agents/xctest 加载它的包。

Xcode 7 之前的版本,它会在运行任何测试之前加载所有 引用的测试包。这可以看出(对于那些关心的人),通过反汇编 Xcode 6.4 的二进制文件,可以看到符号 -[XCTestTool runTestFromBundle:] 的相关部分。

在 Xcode 7 版本的 xctest 中,您可以看到它延迟加载测试包,直到 XCTestSuite 运行实际测试,在实际的 XCTest 框架,可以在符号 __XCTestMain 中看到,它仅在测试的主机应用程序设置后调用。

因为这些在内部调用的顺序发生了变化,所以调用测试的 +load 方法的方式是不同的。没有对 objective-c-runtime 的内部结构进行任何更改。

如果你想在你的应用程序中解决这个问题,你可以做一些事情。首先,您可以使用 +[NSBundle bundleWithPath:] 手动加载您的包,并在其上调用 -load

您还可以将您的测试目标链接回您的测试主机应用程序(我希望您使用的是独立的测试主机而不是您的主应用程序!),这将使其在 xctest 加载主机应用程序时自动加载。

我不会认为这是一个错误,它只是 XCTest 的一个实现细节。

资料来源:过去 3 天只是为了完全不相关的原因反汇编 xctest

关于ios - 在 Xcode 7 中更改了 +load 方法顺序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32639263/

相关文章:

ios - 如何获得 Xcode UI 测试中的授权部分

ios - 如何检查 UILabel 是否为空并将内容附加到标签?

iphone - 动画 UIButton 状态变化

iphone - UICollectionView布局

ios - 如何使用 kCIInputBoostShadowAmountKey?

c# - 替换 "sealed"类的实现

Android 应用程序测试 - 如何使用真实数据库测试 DAO 层?

ios - 如何在 swift 4 中关闭 UISearchBar touchesBegan 上的键盘

ios - 推送通知配置文件

ios - 使用 UIActivityViewController 发送的 UIImage 的控制文件名