是否禁止协议类型的 inout 变量?
Are inout variables of protocol type prohibited?
以下代码:
protocol SomeProtocol {}
class SomeClass: SomeProtocol {}
private func doSomethingWith(inout someVar: SomeProtocol) {}
private var someGlobalVar = SomeClass() // inferring SomeClass's type
doSomethingWith(&someGlobalVar)
产生以下错误:
Cannot invoke 'doSomethingWith' with an argument list of type '(inout SomeClass)'
将倒数第二行更改为 private var someGlobalVar: SomeProtocol = SomeClass()
可解决错误。
主题
当您在声明时将 SomeClass
实例分配给变量时,变量类型被推断为 SomeClass
。和写一样
private var someGlobalVar: SomeClass = SomeClass()
但是,当传递给 inout
参数时,该函数可以将另一个实例分配给该变量,例如
private func doSomethingWith(inout someVar: SomeProtocol) {
someVar = OtherClass()
}
现在您的类型不匹配。您看到的错误是 Swift 阻止您遇到类似的问题。
换句话说:如果您将一个变量传递给一个函数,并且您知道该函数可以将采用 SomeProtocol
的任何实例分配给该变量,那么您必须使用一个可以实际容纳任何实例的变量采用 SomeProtocol
:
private var someGlobalVar: SomeProtocol
除了@Sulthan说的,还有两种可能的解决方案,
取决于你的功能需要做什么。
您可以使函数 通用:
func doSomethingWith<T : SomeProtocol>(inout someVar: T) {}
现在您可以传递任何 class 符合协议的实例:
var someGlobalVar = SomeClass()
doSomethingWith(&someGlobalVar)
如果您只使用 class 的实例并且只使用函数
修改实例指向的对象的属性,那么您根本不需要 inout 参数,因为 classes 是引用类型。
您只需要将协议标记为 "class protocol":
protocol SomeProtocol : class {
var name : String { get set }
}
class SomeClass: SomeProtocol {
var name : String = ""
}
func doSomethingWith(someVar: SomeProtocol) {
// Modify the object:
someVar.name = "modfied"
}
var someGlobalVar = SomeClass()
doSomethingWith(someGlobalVar)
print(someGlobalVar.name) // "modified"
以下代码:
protocol SomeProtocol {}
class SomeClass: SomeProtocol {}
private func doSomethingWith(inout someVar: SomeProtocol) {}
private var someGlobalVar = SomeClass() // inferring SomeClass's type
doSomethingWith(&someGlobalVar)
产生以下错误:
Cannot invoke 'doSomethingWith' with an argument list of type '(inout SomeClass)'
将倒数第二行更改为 private var someGlobalVar: SomeProtocol = SomeClass()
可解决错误。
主题
当您在声明时将 SomeClass
实例分配给变量时,变量类型被推断为 SomeClass
。和写一样
private var someGlobalVar: SomeClass = SomeClass()
但是,当传递给 inout
参数时,该函数可以将另一个实例分配给该变量,例如
private func doSomethingWith(inout someVar: SomeProtocol) {
someVar = OtherClass()
}
现在您的类型不匹配。您看到的错误是 Swift 阻止您遇到类似的问题。
换句话说:如果您将一个变量传递给一个函数,并且您知道该函数可以将采用 SomeProtocol
的任何实例分配给该变量,那么您必须使用一个可以实际容纳任何实例的变量采用 SomeProtocol
:
private var someGlobalVar: SomeProtocol
除了@Sulthan说的,还有两种可能的解决方案, 取决于你的功能需要做什么。
您可以使函数 通用:
func doSomethingWith<T : SomeProtocol>(inout someVar: T) {}
现在您可以传递任何 class 符合协议的实例:
var someGlobalVar = SomeClass()
doSomethingWith(&someGlobalVar)
如果您只使用 class 的实例并且只使用函数 修改实例指向的对象的属性,那么您根本不需要 inout 参数,因为 classes 是引用类型。 您只需要将协议标记为 "class protocol":
protocol SomeProtocol : class {
var name : String { get set }
}
class SomeClass: SomeProtocol {
var name : String = ""
}
func doSomethingWith(someVar: SomeProtocol) {
// Modify the object:
someVar.name = "modfied"
}
var someGlobalVar = SomeClass()
doSomethingWith(someGlobalVar)
print(someGlobalVar.name) // "modified"