是否禁止协议类型的 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"