How to define initializers in a protocol extension?

后端 未结 4 481
佛祖请我去吃肉
佛祖请我去吃肉 2021-01-30 20:06
protocol Car {
     var wheels : Int { get set}

     init(wheels: Int)

}

extension Car {
    init(wheels: Int) {
        self.wheels = wheels
    }
}
<
4条回答
  •  长情又很酷
    2021-01-30 20:53

    As you can see this doesn't work under these circumstances because when compiling, one has to make sure that all properties are initialized before using the struct/enum/class.

    You can make another initializer a requirement so the compiler knows that all properties are initialized:

    protocol Car {
        var wheels : Int { get set }
        // make another initializer
        // (which you probably don't want to provide a default implementation)
        // a protocol requirement. Care about recursive initializer calls :)
        init()
        init(wheels: Int)
    
    }
    
    extension Car {
        // now you can provide a default implementation
        init(wheels: Int) {
            self.init()
            self.wheels = wheels
        }
    }
    
    // example usage
    
    // mark as final
    final class HoverCar: Car {
        var wheels = 0
        init() {}
    }
    
    let drivableHoverCar = HoverCar(wheels: 4)
    drivableHoverCar.wheels // 4
    

    As of Xcode 7.3 beta 1 it works with structs as expected but not with classes since if they are not final the init(wheels: Int) in the protocol is a required init and it can be overridden therefore it cannot be added through an extension. Workaround (as the complier suggests): Make the class final.

    Another workaround (in depth; without final class)

    To work with classes without making them final you can also drop the init(wheels: Int) requirement in the protocol. It seems that it behaves no different than before but consider this code:

    protocol Car {
        var wheels : Int { get set }
        init()
        // there is no   init(wheels: Int)
    }
    
    extension Car {
        init(wheels: Int) {
            self.init()
            print("Extension")
            self.wheels = wheels
        }
    }
    
    class HoverCar: Car {
        var wheels = 0
        required init() {}
        init(wheels: Int) {
            print("HoverCar")
            self.wheels = wheels
        }
    }
    
    // prints "HoverCar"
    let drivableHoverCar = HoverCar(wheels: 4)
    
    func makeNewCarFromCar(car: T) -> T {
        return T(wheels: car.wheels)
    }
    
    // prints "Extension"
    makeNewCarFromCar(drivableHoverCar)
    

    So if you make a Car from a generic context where the type on which you call init is only to be known as Car the extension initializer is called even though an initializer is defined in HoverCar. This only occurs because there is no init(wheels: Int) requirement in the protocol.

    If you add it you have the former problem with declaring the class as final but now it prints two times "HoverCar". Either way the second problem probably never occurs so it might be a better solution.

    Sidenote: If I have made some mistakes (code, language, grammar,...) you're welcome to correct me :)

提交回复
热议问题