-
什么是单元测试
单元测试,就是测试代码"单元"的功能,以确保在任何可能的条件下达到预期目的的一种测试方法.单元(Unit)是代码中一个可测试的逻辑部分. 单元测试可以帮助开发者找到错误和崩溃原因,这也是苹果拒绝上架的首要原因(crash了还上个毛架..)!
测试方法应该要能够响应所有类型的输入,包括有效输入和无效输入的情况,以确保单元能够正常运行,可以这么理解,在正常操作下要能获得我想要的结果,在异常情况(空值,缺少参数..)等一些条件下不反回对象,甚至能够对错误进行处理和回应.无论开发者对单元进行了什么更改,现有的测试方法都应该能够成功运行,而新加的测试也应该要成功运行.所以测试方法很关键!
But.很多开发者对单元测试不太感兴趣,虽然它十分整洁,可以验证许许多多的问题,但是创建,维护却需要花时间.如果要覆盖所有的功能和使用场景的话.意味着投入精力会更多一点.
小结: 测试可以增加项目的稳定性,减少错误的发生.一个良好的测试可以极高地提升用户的满意度.
-
测试基础概念
Xcode使用XCTest作为单元测试框架,在Xcode5之前,使用的是一个名为OCUnit的开源测试框架. XCTest就是对OCUnit的替代品,能够更好地与XCode协作.
单元测试的概念中有四个层级:- 测试套件(Test suite): 测试套件是项目中所有测试的集合,在Xcode中,测试套件作为一个独立的对象存在.
-
测试用类例(Test case classes): 测试功能是存放在类当中的,每个测试的例类通常是对应一个单独类来进行测试.比如: 对Login类的测试 ,应该由LoginTests类来完成,所有的单元测试类都必须要继承XCTestCase类.
-
** 测试用例方法():** 测试用例类包含多个方法,用来测试类的各种功能.
-
断言(Assertions): 断言用于检查结果是否符合预期,如果不符合,断言则会失败,并抛出失败的原因.(调试神器)
![](https://img.haomeiwen.com/i1315383/bbccd2cc51915090.png)
创建项目的时候勾选 Include Unit Tests
成功创建之后,打开项目结构.系统帮我们生成了一个 "项目名+Tests"的文件夹.已经囊括了单元测试的 .m文件.
![](https://img.haomeiwen.com/i1315383/7416cdaf43d34bf9.png)
分析下 .m的结构
<code>
#import <XCTest/XCTest.h>
#import "Person.h"
@interface UnitTestTests : XCTestCase
@end
@implementation UnitTestTests
- (void)setUp {
[super setUp];
// Put setup code here. This method is called before the invocation of each test method in the class.
}
- (void)tearDown {
[super tearDown];
// Put setup code here. This method is called before the invocation of each test method in the class.
}
- (void)testExample {
}
- (void)testPerformanceExample {
[self measureBlock:^{
}];
}
</code>
说说这里面的几个方法.
测试用例类 包含了 setUp 和 tearDown 方法, 这两个方法不属于测试用例方法,他们是测试用例类的初始化和析构方法.setUp里面存放的是所有对象的设置代码,而tearDown里面存放的是诸如关闭文件,取消网络请求等清理活动的代码.Xcode会依次调用 setUp,某个测试用例方法和tearDown方法.如果有多个测试方法的话,那么setUp和tearDown会在每调用一次测试方法的过程中调用一次!
testExample是示例测试方法
testPerformanceExample是性能测试的示例方法.
对于Xcode来说,测试分两种:
a. 功能测试 :功能不符合预期会报错.
b. 性能测试 :性能测试需要设置基准线,一旦发现测试结果低于基准线,或者超出最大标准差(STDDEV)限制,就会报告一个错误.
PS:这里就不对性能测试做其他说明了,有兴趣可以自行搜集资料.
注意: 自定义的测试方法必须以 test开始,这样Xcode才能找到. (左边有图标则表明方法有效)
![](https://img.haomeiwen.com/i1315383/c0175a95a054b4a0.png)
运行测试
当指向测试类和测试方法的时候,会出现一个按钮,我们可以运行所有的测试,也可以运行某一个独立的测试方法.测试结束后,Xcode将会返回成功或者是失败结果
![](https://img.haomeiwen.com/i1315383/1c84e0ab924b50fe.png)
当然,我们也可以选中我们想要运行的测试类,然后选择菜单栏的 product->performAction -> Run Test Methods来运行选中的测试类.如果你只选择了一个测试类的话,那么相应的选项会变成 Run "测试方法名".
![](https://img.haomeiwen.com/i1315383/89580cc9f4087590.png)
Xcode将自动编译并运行应用程序,然后执行测试操作,测试操作完毕后,Xcode会退出应用,短暂的显示测试结果,同时测试结果会以相应的图标表示测试是否成功!
在修改好我们发现的问题之后,单击失败测试上的运行按钮,或者 选择product->performAction -> TestAgian
回到我们的例子.
![](https://img.haomeiwen.com/i1315383/8c90c8e4fc124039.png)
这是.m的实现文件
![](https://img.haomeiwen.com/i1315383/ca7bfe989bee7e31.png)
简单的给person赋值了一个name,age.
目前没有对age做任何的判断.意味着不论是给age 赋值任何 int类型的 值 都能够成功!这当然不符合我们的预期.
我们在测试类里面可以开始动我们的方法了.
// UnitTestTests.m
// UnitTestTests
//
// Created by uncle-R on 16/7/5.
// Copyright © 2016年 uncle-R. All rights reserved.
//
#import <XCTest/XCTest.h>
#import "Person.h"
@interface UnitTestTests : XCTestCase
@property (nonatomic, strong) Person *p1;
@end
@implementation UnitTestTests
- (void)setUp {
[super setUp];
//给p1.赋值一个 -1的age;
self.p1 = [[Person alloc]initWithName:@"小明" andAge:-1];
// Put setup code here. This method is called before the invocation of each test method in the class.
}
- (void)tearDown {
// Put teardown code here. This method is called after the invocation of each test method in the class.
[super tearDown];
}
- (void)testAge{
//我们想要的结果是年龄不管怎样都必须大于0.
XCTAssert(self.p1.age > 0 ,@"年龄必须大于0");
}
- (void)testExample {
// This is an example of a functional test case.
// Use XCTAssert and related functions to verify your tests produce the correct results.
//[self ageTest];
}
- (void)testPerformanceExample {
// This is an example of a performance test case.
[self measureBlock:^{
// Put the code you want to measure the time of here.
}];
}
结果如下
![](https://img.haomeiwen.com/i1315383/d15da661f31e9cba.png)
可见测试结果失败.我们必须在person类的修改.
#import "Person.h"
@implementation Person
-(instancetype)initWithName:(NSString*)name andAge:(int)age{
if (self = [super init]) {
//对小于0的年龄做处理
if(age <= 0) age = 1;
self.age = age;
self.name = name;
}
return self;
}
@end
在运行一次测试.
![](https://img.haomeiwen.com/i1315383/683cca004f11e7de.png)
我们成功的通过了测试.
-
再谈功能测试
功能测试的核心在于 "断言",上面的例子不难看出,测试结果取决于断言是否成功.
功能测试大致可以分如下几种:
1. 基础测试
2. 布尔测试
3.相等测试
4.空值测试
5.无条件失败
6.测试实例
断言:
|测试名称| 断言 | 特性 |
|: ---- :|:------:| ---- ---:|
| 基础测试 | XCTAssert | 最基础的断言,表达式为假则测试失败|
| 布尔测试 |XCTAssertTure, XCTAssertFalse | 基础测试的扩展,当表达式结果和布尔测试不匹配时则失败.
| 相等测试 | XCTAsserEqual,XCTAssertEqualObjects等 | 两个表达式不相等则测试失败 |
|空值测试 |XCTAssertNotNil |如果表达式为空则测试失败|
| 无条件失败| XCTFail | 总是测试失败(运行过这段代码就失败)|
虽然所有的测试都可以使用XCTAssert,但是Xcode还是提供了一些更有效的宏.当没有更好代替的情况下,才使用XCTAssert
格式:
** XCTAssert(表达式,消息)**
注意一下, 无条件失败通常用在控制流中,比如一个if esle结构中.if 里面都是正常的结果,else 正常情况下压根就不会进去,我们这时候可以往else丢一个 XCTFail进去,立马就能检测出异常.
当测试失败的时候
在发现测试失败的时候,先检查自己测试方法是不是存在问题,因为并不是所有逻辑一开始都想通的,因为有些时候测试失败并不仅仅是测试对象属性为nil,或者方法出问题,还可能是因为测试方法存在某些不达标的地方.
因此在测试之前,最好先想清楚测试的目的,再检查自己测试的过程是否符合要求.
tip: 当断言过多的时候要看仔细, 一个Not 就能毁掉整个测试.
网友评论