In-Out(inout关键字)参数

默认情况下,函数参数默认是常量,试图从函数体中去改变一个函数的参数值会报编译错误。如果希望函数修改参数值,并在函数调用结束后仍然保留。这个时候就需要用到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)

总结:

  1. inout参数的本质是地址传递 (引用传递),不管什么情况都是传入一个地址。
  2. Swift 值类型中,属性的默认行为是不可变的。mutating关键字,用于在结构体或枚举的方法中修改属性。使用mutating修饰的方法(func)在修改属性后更新原始值,而不是返回一个新的副本。
    (mutating关键字只能用于值类型,mutating关键字本质是包装了inout关键字,加上mutating关键字后参数值会变成地址传递。
    类对象是指针,传递的本身就是地址值,所以 mutating关键字对类是透明的,加不加效果都一样。)



 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值