协议数组元素是否通过值或引用传递?



我知道Swift中struct是按值传递的,类是按引用传递的。

我想知道我是否创建了一个数组存储提供协议的元素。这些元素是通过值还是引用传递的?

它是基于模型的定义一个类或结构?

class ClassA: ProtocolA {
// something
}
struct StructA: ProtocolA {
// something
}
var arr: [ProtocolA] = [ClassA(), StructA()]
exampleFunction(arr[0]) // is it passed by reference
exampleFunction(arr[1]) // is it passed by value

无论如何声明数组,类都是通过引用传递的,结构是通过值传递的。这可以用下面的例子来演示:

protocol ProtocolA {
var title: String { get set}
}
class ClassA: ProtocolA {
var title = "ClassA"
}
struct StructA: ProtocolA {
var title = "StructA"
}
var arr: [ProtocolA] = [ClassA(), StructA()]
print(arr[0].title) // ClassA
print(arr[1].title) // StructA
func exampleFunction(_ obj: ProtocolA) {
var obj = obj // Create local mutable variable
obj.title = obj.title + "!!!"
}
exampleFunction(arr[0])
exampleFunction(arr[1])
print(arr[0].title) // ClassA!!!
print(arr[1].title) // StructA

当您将变量存储为协议类型时,编译器将其视为值类型来处理。因此,如果exampleFunction接受ProtocolA类型的输入参数,为了使您能够改变输入参数的属性,您需要将其声明为inout

这并不意味着值必须按值传递,它只是意味着编译器不知道输入参数是值类型还是引用类型,因此在编译时将其作为值类型处理。

如果您希望能够将协议变量作为引用类型处理,则需要将协议类绑定。protocol ProtocolA: class {}

协议应该被视为值类型,因为您需要通过将其定义为符合AnyObject(所有类都符合)来显式地告诉编译器它是引用类型

所以如果你有

protocol ProtocolB: AnyObject

则任何符合协议b的类型将通过引用发送,否则不发送。

下面是一个简化的例子

protocol ProtocolA {
var x: Int { get set }
}
protocol ProtocolB: AnyObject {
var y: Int { get set }
}
class ClassA: ProtocolA, ProtocolB {
var x = 0
var y = 0
}
func exampleFunction(_ object: ProtocolA) {
object.x += 2 // <-- This will generate a compilation error
}
func exampleFunction(_ object: ProtocolB) {
object.y += 2 // This is fine
}

最新更新