Xcode 9.0 Beta:调用Type.init()

时间:2017-06-09 21:47:38

标签: ios swift xcode swift3 xcode9-beta

我刚刚升级到Xcode 9.0 Beta,现在我的应用程序在启动时崩溃了。

2017-06-09 14:35:18.817213-0700 recharge-consumer-ios[13524:1720597] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '*** +[recharge_consumer_ios.SignedOutContainerViewController<0x105c691b8> init]: cannot init a class object.'
*** First throw call stack:
(
0   CoreFoundation                      0x000000010caf4f6b __exceptionPreprocess + 171
1   libobjc.A.dylib                     0x000000010ba3e121 objc_exception_throw + 48
2   CoreFoundation                      0x000000010cb7f6ff +[NSObject(NSObject) init] + 127
3   recharge-consumer-ios               0x0000000105322223 _T021recharge_consumer_ios22RechargeViewControllerCACycfCTD + 19

失败的呼叫是(简化):

func setAndRefreshChildViewControllersWithTypes(_ types: [MyViewControllerProtocol.Type]) {
    for type in types {
        let controller = type.init() as! UIViewController // This is what crashes
    }
    ...
}

MyViewControllerProtocol的位置:

protocol MyViewControllerProtocol {
    // some other fields here too
    init()
}

失败的实例看起来像:

class SignedOutContainerViewController: MyViewController {

    required init() {
        super.init(nibName: "SignedOutContainerViewController")
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
}

其中MyViewController是:

class MyViewController: UIViewController, MyViewControllerProtocol {
    required init() {
        fatalError("init has not been implemented")
    }

    init(nibName nibNameOrNil: String?) {
        super.init(nibName: nibNameOrNil, bundle: nil)
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
}

知道出了什么问题,还是想看看还有什么地方? Xcode 9.0 Beta(9M136h),swift 3。

2 个答案:

答案 0 :(得分:1)

我想跟进,因为接受的答案对我不起作用。所有这些都是在操场上使用17/12/17的Xcode的swift 4.0开发快照工具链进行测试的。

这将因运行时异常而失败

import Foundation

public protocol MyProtocol {
    init()
}

@objc // This doesn't work
public class MyClass: NSObject, MyProtocol {

    @objc // This doesn't work either
    required public override init() {
        super.init()
    }
}

let anonymousType: AnyClass = MyClass.self

let metaType: MyProtocol.Type = anonymousType as! MyProtocol.Type
let instance = metaType.init()

对我而言,它并不像使用@objc标记类或方法那么简单。我找到了两个可以为我工作的解决方案。

  1. 标记班级最终

    import Foundation
    
    public protocol MyProtocol {
        init()
    }
    
    final public class MyClass: NSObject, MyProtocol {
    
        required public override init() {
            super.init()
        }
    }
    
    let anonymousType: AnyClass = MyClass.self
    
    let metaType: MyProtocol.Type = anonymousType as! MyProtocol.Type
    let instance = metaType.init()
    
  2. 标记协议@objc

    import Foundation
    
    @objc
    public protocol MyProtocol {
        init()
    }
    
    public class MyClass: NSObject, MyProtocol {
    
        required public override init() {
            super.init()
        }
    }
    
    let anonymousType: AnyClass = MyClass.self
    
    let metaType: MyProtocol.Type = anonymousType as! MyProtocol.Type
    let instance = metaType.init()
    

答案 1 :(得分:0)

将@objc用于崩溃的方法