我正在尝试使用 Storyboard快速测试 UITableViewController 的子类。我能够获得对 View Controller 的引用并记录它,但我无法将它类型转换为我尝试测试的类,因此我无法访问方法、属性等。

获取我的类实例的正确方法是什么?

测试场景:

import XCTest
import UIKit
class GameListControllerTest: XCTestCase {
    var storyboard = UIStoryboard(name: "Main", bundle: nil)
    var _sut: AnyObject?
    var sut: AnyObject {
        if(_sut?) {
            return _sut!
        }
        _sut = storyboard.instantiateViewControllerWithIdentifier("GameListController")
        return _sut!
    }
    override func setUp() {
        super.setUp()
        println()
        println("=================================")
        println("sut  \(sut)")
        println("view  \(sut.view)")
        println("=================================")

    }
    func testInstance() {
        var vc1 = sut as UITableViewController
        var vc2 = sut as? GameListController
        println()
        println("=================================")
        println("VC1  \(vc1)")
        println("VC2  \(vc2)")
        println("=================================")
    }
}

输出:
XCTestOutputBarrierTest Suite '_TtC15mytabletopTests22GameListControllerTest' started at 2014-06-16 06:13:30 +0000
XCTestOutputBarrierTest Case '-[_TtC15mytabletopTests22GameListControllerTest testInstance]' started.
XCTestOutputBarrier
=================================
sut  <_TtC10mytabletop18GameListController: 0xb338d50>
view  <UITableView: 0xc033800; frame = (0 0; 320 568); clipsToBounds = YES; autoresize = W+H; gestureRecognizers = <NSArray: 0xb344960>; layer = <CALayer: 0xb33c7d0>; contentOffset: {0, 0}; contentSize: {480, 0}>
=================================

=================================
VC1  <_TtC10mytabletop18GameListController: 0xb338d50>
VC2  nil
=================================
Test Case '-[_TtC15mytabletopTests22GameListControllerTest testInstance]' passed (0.007 seconds).
XCTestOutputBarrierTest Suite '_TtC15mytabletopTests22GameListControllerTest' passed at 2014-06-16 06:13:30 +0000.
     Executed 1 test, with 0 failures (0 unexpected) in 0.007 (0.010) seconds

如果我尝试强制类型转换 ( sut as GameListController ),我会收到运行时异常。

最佳答案

这就是正在发生的事情。我花了好几天才弄明白,但我的问题是我的类是如何导出到目标的:

这导致我的类(class)有两个二进制副本,一个在应用程序目标中,另一个在测试目标中。如果我们更加关注日志,我们可能会注意到:

Test Case '-[_TtC15mytabletopTests22GameListControllerTest testInstance]'

以上是测试方法 testInstance ,它是 mytabletopTests 执行上下文的一部分。现在让我们看一下从 Storyboard 中提取的实例:
sut  <_TtC10mytabletop18GameListController: 0xb338d50>

这反过来又在 mytabletop 上下文中运行。这解释了为什么测试无法类型转换为 GameListController 。它知道的 GameListController 是在测试目标中编译的那个。

由于从测试目标中删除类会使我的测试用例不知道该类,现在我需要将我的应用程序目标导入到测试用例中:
import XCTest
import UIKit
import mytabletop // LINE ADDED

class GameListControllerTest: XCTestCase {

现在,测试可以访问的唯一 GameListController 与 Storyboard正在实例化的代码相同,我终于能够对其进行类型转换。这是新的测试用例:
import XCTest
import UIKit
import mytabletop

class GameListControllerTest: XCTestCase {
    let sut: GameListController = UIStoryboard(name: "Main", bundle: nil).instantiateViewControllerWithIdentifier("GameListController") as GameListController

    override func setUp() {
        super.setUp()
        UIApplication.sharedApplication().keyWindow.rootViewController = sut
        XCTAssertNotNil(sut.view)
    }

    func testInstance() {
        XCTAssertNotNil(sut)
        XCTAssertNotNil(sut.tableView) // UITableViewController property
        XCTAssertNotNil(sut.store) // instance property
        XCTAssertNotNil(sut.someButton) // outlet
    }
}

现在,我可以在实例初始化期间正确地进行类型转换(滚动以查看 as GameListController )。为了强制所有 socket 正确绑定(bind), subview 相应地呈现到正在运行的设备上,我们可以使 View Controller 成为应用程序的 rootViewController 并从中拉出 View ,如上面的 setUp 函数所示。甚至 myCustomOutlet 现在也能正常工作。

关于ios - 如何在 Swift 中对子类 UIViewController 进行类型转换?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/24237609/

10-09 04:21