在iOS上单元测试UIViewController.viewDidLoad()
方法时,每个人最喜欢的注入依赖关系的方式是什么?
鉴于我的实施是:
class MyViewController: UIViewController {
var service: Service = Service()
override func viewDidLoad() {
super.viewDidLoad()
service.load()
}
}
我的测试课就是这样的:
class MyViewController Tests: XCTestCase {
var vc: MyViewController!
var serviceMock = ServiceMock()
override func setUp() {
super.setUp()
let storyboard = UIStoryboard(name: "Main", bundle: nil)
vc = storyboard.instantiateViewController(withIdentifier: "MyViewController") as! MyViewController
vc.service = serviceMock
}
}
func testThatServiceIsCalled() {
XCTAssertTrue(serviceMock.loadCalled)
}
这里显而易见的问题是,在我实例化viewController时调用了viewDidLoad()
,并且由于模拟没有正确注入,测试失败。
任何解决方案?
答案 0 :(得分:1)
为什么需要使用故事板来创建视图控制器?
这不会起作用吗?
func testThatServiceIsCalled() {
let vc = MyViewController()
vc.service = serviceMock
vc.viewDidLoad()
XCTAssertTrue(serviceMock.loadCalled)
}
答案 1 :(得分:-1)
TL; DR您不应该像这样对与UI相关的方法进行单元测试。
控制器和视图不适合单元测试,原因很简单,因为它们通常涉及在屏幕上反映的操作,因此(非常)难以测试。对于MVC的这两个组件,快照测试或UI自动化测试是更好的选择。
编写良好的Controller不包含业务逻辑,它将UI内容委托给View,并将业务操作委托给其Model。考虑到这一点,它应该是一个可以进行单元测试的模型。
现在,如果你真的想断言你的服务是用适当的方法调用的,你可以将你的VC设置为root视图控制器,这应该确保loadView
方法获得调用者,应该触发viewDidLoad
一个:
func testThatServiceIsCalled() {
let window = UIWindow(frame: UIScreen.main.bounds)
window.makeKeyAndVisible()
window.rootViewController = vc
XCTAssertTrue(serviceMock.loadCalled)
}