我发现 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
出于某种原因是某种特殊情况吗?
为什么我要问这个
- 教育目的。
- 我曾经在单元测试目标内的类别中执行方法调配。现在,当调用顺序改变时,
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/