我在Swift游乐场中有以下示例,试图在Swift中实现一个复制构造函数:
class Shape : NSObject {
var color : String
override init() {
color = "Red"
}
init(copyFrom: Shape) {
color = copyFrom.color
}
}
class Square : Shape {
var length : Double
override init() {
super.init()
length = 10.0
}
init(copyFrom: Square) { /* Compilation error here! */
super.init(copyFrom: copyFrom)
length = copyFrom.length
}
}
let s : Square = Square() // {{color "Red"} length 10.0}
let copy = Square(copyFrom: s) // {{color "Red"} length 10.0}
s.color = "Blue" // {{color "Blue"} length 10.0}
s // {{color "Blue"} length 10.0}
copy // {{color "Red"} length 10.0}
问题在于,它实际上并没有以当前形式进行编译。在init(copyFrom: Square)
子类中的Square
方法上,报告了此错误:
Overriding method with selector 'initWithCopyFrom:' has incompatible type '(Square) -> Square'
这个问题会有意义如果它不是构造函数,就好像它是一个普通的func
,你可能会传入一个类型这在超类中是预期的,但是在子类中被覆盖了更严格的限制:
let mySquare : Shape = Square() // Note the var is a SHAPE
mySquare.someShapeMethod("Test") // If Square overrides someShapeMethod() to expect Int, compiler errors out to protect us here.
但事实上它是构造函数让我相信我应该能够覆盖它并提供不同的方法签名,因为它在编译时是绝对已知的对象的类型是什么。
如果我将Shape
更改为不再延长NSObject
,则此问题会消失。但是,由于包含现有的Objective-C代码,因此需要扩展NSObject
。
如何更新我的复制构造函数以允许Shape
知道它从Shape
进行复制,并允许Square
知道它的复制来自Square
?
答案 0 :(得分:22)
init(copyFrom: Square)
是init(copyFrom: Shape)
的重载,而不是重写。我的意思是它们是不相关的方法,因为它们接受不同的类型。在斯威夫特这是可以接受的。在ObjC,这是非法的。 ObjC没有重载。
Swift初始化程序不会自动继承。因此,在Swift中,您无法尝试将随机Shape
复制为Square
。初始化程序不可用。但是在ObjC中,初始化器做会自动继承(并且你无法阻止它们这样做)。因此,如果您有方法initWithCopyFrom:(*Shape)
,则要求每个子类都愿意接受它。这意味着您可以(在ObjC中)尝试创建Circle的副本作为Square。那当然是胡说八道。
如果这是NSObject
子类,则应使用NSCopying
。以下是你将如何做到这一点:
import Foundation
class Shape : NSObject, NSCopying { // <== Note NSCopying
var color : String
required override init() { // <== Need "required" because we need to call dynamicType() below
color = "Red"
}
func copyWithZone(zone: NSZone) -> AnyObject { // <== NSCopying
// *** Construct "one of my current class". This is why init() is a required initializer
let theCopy = self.dynamicType()
theCopy.color = self.color
return theCopy
}
}
class Square : Shape {
var length : Double
required init() {
length = 10.0
super.init()
}
override func copyWithZone(zone: NSZone) -> AnyObject { // <== NSCopying
let theCopy = super.copyWithZone(zone) as Square // <== Need casting since it returns AnyObject
theCopy.length = self.length
return theCopy
}
}
let s = Square() // {{color "Red"} length 10.0}
let copy = s.copy() as Square // {{color "Red"} length 10.0} // <== copy() requires a cast
s.color = "Blue" // {{color "Blue"} length 10.0}
s // {{color "Blue"} length 10.0}
copy // {{color "Red"}
Swift 3
class Shape: NSObject, NSCopying {
required override init() {
super.init()
}
func copy(with zone: NSZone? = nil) -> Any {
let copy = type(of: self).init()
return copy
}
}
class Square: Shape {
required override init() {
super.init()
}
func copy(with zone: NSZone? = nil) -> Any {
let copy = super.copy(with: zone) as! Square
copy.foo = self.foo
......
return copy
}
}
答案 1 :(得分:2)
最简单的方法就是将子类初始化程序的名称更改为init(copyFromSquare: Square)
,将Square
方法保留为init(copyFrom: Shape)
方法(因为您已通过继承而签订合同) Shape
)。
您当然可以覆盖init(copyFrom: Shape)
,并测试copyFrom
是否为Square
,在这种情况下,您采取一种行动(设定长度),否则不会。
另请注意,您需要在之前设置self.length
来调用超级。
class Shape : NSObject {
var color : String
override init() {
color = "Red"
}
init(copyFrom: Shape) {
color = copyFrom.color
}
}
class Square : Shape {
var length : Double
override init() {
self.length = 10.0
super.init()
}
override init(copyFrom: Shape) {
if copyFrom is Square {
self.length = (copyFrom as Square).length
} else {
self.length = 10.0 // default
}
super.init(copyFrom: copyFrom)
}
}