创建单元测试项目
创建项目时直接勾选Include Unit Tests
,项目会生成用于测试的.m
文件和plist
文件,Include UI Tests
则是用于UI
测试的,后面会介绍。
万一我忘了勾选怎么办呢?可以有其他方式创建
File -> new -> target -> iOS -> iOS Unit Testing Bundle
。
单元测试方法介绍
找到系统单元测试Testes
文件夹中.m
文件看中会到看到几个方法,我们来看下这个几个方法是什么时候调用和他们各种的作用:
- (void)setUp {
[super setUp];
//初始化的代码,在测试方法调用之前调用
}
- (void)tearDown {
// 释放测试用例的资源代码,这个方法会每个测试用例执行后调用
[super tearDown];
}
- (void)testExample {
// 测试用例的例子,注意测试用例一定要test开头
}
- (void)testPerformanceExample {
// 测试性能例子
[self measureBlock:^{
// 需要测试性能的代码
}];
}
常用的断言介绍
单元测试主要使用XCT
断言来测试。
XCTFail(format…) 生成一个失败的测试;
XCTAssertNil(a1, format...)为空判断,a1为空时通过,反之不通过;
XCTAssertNotNil(a1, format…)不为空判断,a1不为空时通过,反之不通过;
XCTAssert(expression, format...)当expression求值为TRUE时通过;
XCTAssertTrue(expression, format...)当expression求值为TRUE时通过;
XCTAssertFalse(expression, format...)当expression求值为False时通过;
XCTAssertEqualObjects(a1, a2, format...)判断相等,[a1 isEqual:a2]值为TRUE时通过,其中一个不为空时,不通过;
XCTAssertNotEqualObjects(a1, a2, format...)判断不等,[a1 isEqual:a2]值为False时通过;
XCTAssertEqual(a1, a2, format...)判断相等(当a1和a2是 C语言标量、结构体或联合体时使用, 判断的是变量的地址,如果地址相同则返回TRUE,否则返回NO);
XCTAssertNotEqual(a1, a2, format...)判断不等(当a1和a2是 C语言标量、结构体或联合体时使用);
XCTAssertEqualWithAccuracy(a1, a2, accuracy, format...)判断相等,(double或float类型)提供一个误差范围,当在误差范围(+/-accuracy)以内相等时通过测试;
XCTAssertNotEqualWithAccuracy(a1, a2, accuracy, format...) 判断不等,(double或float类型)提供一个误差范围,当在误差范围以内不等时通过测试;
XCTAssertThrows(expression, format...)异常测试,当expression发生异常时通过;反之不通过;(很变态) XCTAssertThrowsSpecific(expression, specificException, format...) 异常测试,当expression发生specificException异常时通过;反之发生其他异常或不发生异常均不通过;
XCTAssertThrowsSpecificNamed(expression, specificException, exception_name, format...)异常测试,当expression发生具体异常、具体异常名称的异常时通过测试,反之不通过;
XCTAssertNoThrow(expression, format…)异常测试,当expression没有发生异常时通过测试;
XCTAssertNoThrowSpecific(expression, specificException, format...)异常测试,当expression没有发生具体异常、具体异常名称的异常时通过测试,反之不通过;
XCTAssertNoThrowSpecificNamed(expression, specificException, exception_name, format...)异常测试,当expression没有发生具体异常、具体异常名称的异常时通过测试,反之不通过
使用
测试公有方法
比如有一个User
类:
@interface User : NSObject
@property (nonatomic, copy) NSString *username;
@property (nonatomic, copy) NSString *password;
- (NSInteger)getCode;
@end
@implementation User
// 模拟返回数据
- (NSInteger)getCode {
return 100;
}
@end
现在需要测试getCode
方法:
@interface UnitTestsDemoTests : XCTestCase
@property (nonatomic, strong) User *user;
@end
@implementation UnitTestsDemoTests
- (void)setUp {
[super setUp];
self.user = [[User alloc] init];
}
- (void)tearDown {
self.user = nil;
[super tearDown];
}
- (void)testUser {
XCTAssertEqual([self.user getCode] , 100, @"登录码错误");
}
- (void)testPerformanceExample {
// This is an example of a performance test case.
[self measureBlock:^{
// Put the code you want to measure the time of here.
}];
}
@end
测试方法写好后,你可以点击测试方法前的菱形标记测试单一方法,也可以comand+u
测试全部。
当测试通过时,菱形标记会打钩:
我改一下判断值:
- (void)testUser {
XCTAssertEqual([self.user getCode] , 200, @"登录码错误");
}
测试不通过时,菱形标记会打叉:
测试私有方法
测试私有方法就不能通过保存对象来测试,要通过类别来测试。
给User
添加一个私有方法,判断username
是否为手机号码:
- (BOOL)usernameIsPhoneNumber {
// validateCellPhoneNumber方法是一个正则表达式判断手机号的方法
if ([User validateCellPhoneNumber:self.username]) {
return YES;
}
return NO;
}
在单元测试Testes
文件夹中.m
文件添加User
的类别:
@interface User (Tests)
- (BOOL)usernameIsPhoneNumber;
@end
然后就可以调用私有方法测试了:
- (void)testUser {
self.user.username = @"15875512345";
// XCTAssertTrue([self.user performSelector:@selector(usernameIsPhoneNumber)], @"用户名不是手机号码");
XCTAssertTrue([self.user usernameIsPhoneNumber], @"用户名不是手机号码");
}
网络请求的测试
iOS9
的http
安全问题:现在进行异步请求的网络测试,由于测试方法主线程执行完就会结束,所以需要设置一下,否则没法查看异步返回结果。在方法结束前设置等待,调回回来的时候再让它继续执行。
定义宏如下:
//waitForExpectationsWithTimeout是等待时间,超过了就不再等待往下执行。
#define WAIT do {\
[self expectationForNotification:@"RSBaseTest" object:nil handler:nil];\
[self waitForExpectationsWithTimeout:30 handler:nil];\
} while (0);
#define NOTIFY \
[[NSNotificationCenter defaultCenter]postNotificationName:@"RSBaseTest" object:nil];
测试网络请求:
- (void)testRequest {
// 1.获得请求管理者
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
[manager GET:@"http://jsonplaceholder.typicode.com/posts" parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
} success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
NSLog(@"responseObject:%@",responseObject);
XCTAssertNotNil(responseObject, @"返回出错");
NOTIFY // 继续执行
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
XCTAssertNil(error, @"请求出错");
NOTIFY // 继续执行
}];
WAIT // 暂停
}
UI测试
用代码写UI
测试比较麻烦,但是苹果在Xcode
中为我们提供了录制的功能。录制是怎么一回事呢?当你打开时这个功能时,测试代码会随着你在设备或模拟器上操作自动创建。这么一来就省事多了。
找到系统单元测试UITestes
文件夹中.m
文件,然后把光标放在方法体内,然后点击红色的那个录制按钮,操作界面,系统会自动为我们添加代码。根据编写的代码写我们的测试。
//XCUIApplication 这是应用的代理,他能够把你的应用启动起来,并且每次都在一个新进程中。
XCUIApplication *app = [[XCUIApplication alloc] init];
//XCUIElement 这是 UI 元素的代理。元素都有类型和唯一标识。可以结合使用来找到元素在哪里,如当前界面上的一个输入框
XCUIElement *usernameTextField = app.textFields[@"username:"];
[usernameTextField tap];
[usernameTextField typeText:@"xiaofei"];
XCUIElement *passwordTextField = app.textFields[@"password:"];
[passwordTextField tap];
[passwordTextField tap];
[passwordTextField typeText:@"12345"];
[[[[[[[app childrenMatchingType:XCUIElementTypeWindow] elementBoundByIndex:0] childrenMatchingType:XCUIElementTypeOther].element childrenMatchingType:XCUIElementTypeOther].element childrenMatchingType:XCUIElementTypeOther].element childrenMatchingType:XCUIElementTypeOther].element tap];
[app.buttons[@"login"] tap];
//登录成功后的控制器的title为loginSuccess,只需判断控制器的title时候一样便可判断登录是否成功
XCTAssertEqualObjects(app.navigationBars.element.identifier, @"loginSuccess");