很长一段时间以来,我一直在努力使用Swift协议和关联类型。我重新开始基本的,以便真正理解出现了什么问题,我在Rob Napier的Swift Protocols中使用了相关类型要求的article TypeErasure,但我仍然没有运气。
找到下面的代码
// An Animal can eat
protocol Animal {
associatedtype Food
func feed(food: Food) -> Void
}
struct AnyAnimal<Food>: Animal {
private let _feed: (Food) -> Void
init<Base: Animal where Food == Base.Food>(_ base: Base) {
_feed = base.feed
}
func feed(food: Food) { _feed(food) }
}
// Kinds of Food
struct Grass {}
struct Cow: Animal {
func feed(food: Grass) { print("moo") }
}
struct Goat: Animal {
func feed(food: Grass) { print("bah") }
}
let grassEaters = [AnyAnimal(Cow()), AnyAnimal(Goat())]
for animal in grassEaters {
animal.feed(Grass())
}
现在我想在协议Animal中给出一个默认实现,如下所示
extension Animal {
func feed(food: Food) -> Void {
print("unknown")
}
}
当我从struct Cow中删除该函数时,我得到的错误消息是Cow不符合Protocol Animal。
这是否意味着你不能同时使用Type Erasures和Default Implementation?有什么方法我可以做TypeErasure并保持默认实现吗?
答案 0 :(得分:6)
问题与类型擦除无关,你会得到相同的结果
如果删除struct AnyAnimal<Food>
定义,则会显示错误消息。
如果您从feed()
删除struct Cow
方法,那么
编译器无法推断关联的类型Food
。所以要么你
在默认实现中使用具体类型:
extension Animal {
func feed(food: Grass) -> Void {
print("unknown")
}
}
struct Cow: Animal {
}
或者使用。为每种类型定义类型别名Food
默认实现:
extension Animal {
func feed(food: Food) -> Void {
print("unknown")
}
}
struct Cow: Animal {
typealias Food = Grass
}
还可以在中定义Food
的默认类型
协议:
protocol Animal {
associatedtype Food = Grass
func feed(food: Food) -> Void
}
extension Animal {
func feed(food: Food) -> Void {
print("unknown")
}
}
struct Cow: Animal {
}