默认情况下,函数参数默认是常量,试图从函数体中去改变一个函数的参数值会报编译错误。如果希望函数修改参数值,并在函数调用结束后仍然保留。这个时候就需要用到inout关键字。
注意事项:
inout关键字只能修饰变量,无法修饰常量,因为常量和字面量不能被修改。
inout参数不能有默认值,可变参数不能标记为inout。
调用函数的时候,应该在变量名前放置&符号表示该变量可以由函数修改。
var variable: Int = 1
func changeNumber(num:inout Int) {
num = 2
print(num)
}
changeNumber(num: &variable) // 2
nout关键字修饰的变量传递过程:
如果实参有物理内存地址,且没有设置属性观察器
答: 直接将实参的内存地址传入函数 (实参进行引用传递)
如果实参是计算属性 或者 设置了属性观察器
答:采取了Copy In Copy Out 的做法
调用该函数时,先复制实参的值,产生副本 [get]
将副本的内存地址传入函数 (副本进行引用传递),在函数内部可以修改副本的值
函数返回后,再将副本的值覆盖实参的值 [set]
var man = person(heigth: 1, width: 1)
{
willSet(newValue){
print("即将 \(newValue) ")
}
didSet (oldValue){
print("已经 \(man) \(oldValue) ")
}
}
func addValue( _ value : inout person)->person
{
value.heigth = 20
value.width = 20
//如果将属性观察关掉,崩溃:因为违反了独占原则:一个地址被别的地方引用后,不可以访问。Swift 需要对变量进行独占访问时才能修改该变量。本质上来说,当一个变量作为 inout 参数或者 mutating 方法中的 self 被修改时,不能通过不同的名称被访问的。
print("新",value,"旧",self.man)
return value
}
//执行
let k = addValue(&man)
print(k)
//打印 因为有属性观察器所以 copy。 执行方法函数结束后再进入属性观察器,再给返回值。
// 新 person(heigth: 20, width: 20) 旧 person(heigth: 1, width: 1)
// 即将 person(heigth: 20, width: 20)
// 已经 person(heigth: 20, width: 20) person(heigth: 1, width: 1)
// person(heigth: 20, width: 20)
总结:
- inout参数的本质是地址传递 (引用传递),不管什么情况都是传入一个地址。
- Swift 值类型中,属性的默认行为是不可变的。mutating关键字,用于在结构体或枚举的方法中修改属性。使用mutating修饰的方法(func)在修改属性后更新原始值,而不是返回一个新的副本。
(mutating关键字只能用于值类型,mutating关键字本质是包装了inout关键字,加上mutating关键字后参数值会变成地址传递。
类对象是指针,传递的本身就是地址值,所以 mutating关键字对类是透明的,加不加效果都一样。)