具有关联类型要求和默认实施的Swift协议

时间:2016-04-23 11:33:27

标签: ios swift generics swift-protocols

很长一段时间以来,我一直在努力使用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并保持默认实现吗?

1 个答案:

答案 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 {
}