【发布时间】:2017-07-19 11:29:31
【问题描述】:
我想根据类型(URL 的一部分,如果你必须知道的话)创建字符串。
考虑这个示例代码:
import Foundation
protocol TestP {
var p: Int { get }
}
protocol TestQ: TestP {
var q: Int { get }
}
struct TestR: TestQ {
var p = 1
var q = 2
}
func toDescription<T: TestP>(type: T.Type) -> String {
switch type {
case is TestR.Type: return "Arrrr"
default: return "unsupported"
}
}
这看起来相当不错;我不需要依赖不安全的措施(字符串),也不需要单独的枚举。
我们来看一些用法示例:
func example1<T: TestP>(val: T) {
print("Processing \(toDescription(type: T.self))")
}
func example2() {
print("Processing \(toDescription(type: TestR.self))")
}
func example3() {
print("Processing \(toDescription(type: TestQ.self))")
}
虽然前两个函数很好(通用版本特别好!),但第三个不能编译:
错误:在参数类型
TestQ.Protocol.Type,TestQ.Protocol不符合预期类型TestP
TestP.Type 和 TestP.Protocol 也不能用作参数。
如何将协议类型传递给(通用)函数?
【问题讨论】:
-
这没有充分的理由编译 - 比如说
TestP有一个static要求。您可以在toDescription中调用type上的该要求——但如果您能够传入TestQ.self,则没有可调用的实现。 -
这是更大限制的一部分(为了防止这些不安全的情况,但在其他情况下充满了完全安全的边缘情况),即protocols don't conform to themselves - 所以你不能使用
TestQ作为符合TestP的类型。 -
@Hamish 我明白了。我想我希望那时能够写
func f<T: TestP>(type: T.Protocol),在这种情况下,编译器可以检查我没有在协议类型上调用静态成员。 (好吧,即使在我编写的版本中,它也可能会阻止我访问静态成员,因为它可能会失败。) -
有趣的事实:如果您切换此类参数
type,模式is TestQ.Type不会引发警告“从不匹配”。 -
如果协议确实符合自己,那么是的,我希望像
func f<T: TestP>(type: T.Protocol)这样的东西可以工作(尽管它会禁止你的前两个示例工作)
标签: swift generics swift-protocols type-bounds