可选绑定、guard else weak unowned

本文深入探讨了Swift语言中的可选绑定、guard语句、defer执行机制以及weak和unowned引用,通过具体代码示例解释了它们的作用、使用场景及优势,帮助读者理解如何有效利用这些特性提高代码质量和效率。
swift的可选绑定和guard的用法

文章出自我的博客:huhansome的博客

这几个关键字放一起也是醉了,本来想放到上一篇一起,想了想放哪里好像都一样。主要是去了解一下这些语句(关键字)的作用以及使用场景及优势。

func sayHello(name:String?) {
    if name == nil {
        print("姓名是空的")
    } else {
        print("hello world" + " " + name!)
    }
}
sayHello(name: nil)

func sayHello1(name:String?) {
    if let tmpName = name {
        print("hello world" + " " + tmpName)
    }
    print("姓名是空的")
}
sayHello1(name: "zhangsan")

func sayHello2(name:String?) {
    guard let tmpName = name  else {
        print("姓名是空的 from guard")
        return
    }
    print("hello world" + " " + tmpName)
}
sayHello2(name: "zhangsan")


//defer 推迟到最后执行
func sayHello3() {
    print("first")
    defer {
        print("defer statement")
    }
    print("last")
}
sayHello3()

结论

可选绑定本质和if else没区别,好处在于会解包数据 ,在之后使用不必要再去解包,guard else 和可选绑定也非常相似,可以在else里面抛异常或者return。 defer推迟到最后执行,这个得到具体场景具体使用。

weak

class Person {
    var name : String?
    var book : Book?
    deinit {
        print("person release")
    }
}

class Book {
    var name : String?
    weak var person : Person?
    deinit {
        print("book release")
    }
}

var person : Person? = Person()
person?.book = Book()
person = nil

person 强引用book,而book弱引用person,所以在person被释放时,book也没有任何强引用而释放,达到解除循环引用的问题,同时weak标记的对象,在释放之后会自动置为nil,避免了野指针问题。

unowned

class Person {
    var name : String?
    var book : Book?
    
    init(name:String?) {
        self.name = name
        self.book = Book(name: "", person: self)
    }
    
    deinit {
        print("person release")
    }
}

class Book {
    var name : String?
    unowned let person : Person
    
    init(name:String?, person:Person) {
        self.name = name
        self.person = person
    }
    
    deinit {
        print("book release")
    }
}

var person : Person? = Person(name: "zhangsan")
person = nil

无主引用和weak一样可以打破循环引用,区别在于无主引用通常都被期望拥有值, 无法在实例被销毁后将无主引用设为nil,因为非可选类型的变量不允许被赋值为nil。具体引用计数请前往这里

Swift 中,`weak` `unowned` 都是用于打破引用循环的关键字,它们的核心作用是避免两个对象之间形成强引用关系,从而导致内存泄漏。然而,两者在行为适用场景上存在显著差异。 ### `weak` 的特点使用场景 `weak` 关键字用于声明一个弱引用,它不会增加对象的引用计数。当对象被释放后,`weak` 引用会自动被设置为 `nil`,因此它只能用于可选类型(Optional)。这种特性使得 `weak` 非常适合用于那些生命周期不确定的对象引用。 一个常见的使用场景是闭包捕获列表中,尤其是在异步操作中。例如,当一个视图控制器发起网络请求并在请求完成后更新 UI 时,如果在请求过程中用户关闭了该视图控制器,`self` 可能已经被释放。此时使用 `[weak self]` 可以安全地访问 `self`,而不会导致崩溃: ```swift someAsyncRequest { [weak self] in guard let self = self else { return } self.updateUI() } ``` ### `unowned` 的特点使用场景 `unowned` 关键字用于声明一个无主引用,与 `weak` 不同的是,`unowned` 引用不会被自动设置为 `nil`。如果对象被释放,`unowned` 引用将变成一个“悬空指针”,如果尝试访问该引用的方法或属性,程序将会崩溃。因此,`unowned` 必须用于非可选类型,并且只有在可以确定对象在访问时一定不会被释放的情况下才能使用。 一个典型的使用场景是在闭包中捕获 `self`,并且可以确保闭包在执行时 `self` 仍然存在。例如,在一个不会被提前释放的对象中执行闭包时,可以使用 `[unowned self]` 来避免额外的可选解包操作: ```swift let closure: () -> Void = { [unowned self] in self.doSomething() } ``` ### `weak` `unowned` 的区别总结 - **自动置 `nil`**:`weak` 引用在对象释放后会自动置为 `nil`,而 `unowned` 引用则不会。 - **是否可选**:`weak` 必须用于可选类型,而 `unowned` 必须用于非可选类型。 - **安全性**:`weak` 更加安全,因为它可以防止访问已释放的对象;而 `unowned` 更加高效,但要求开发者确保对象在访问时仍然有效。 ### 选择 `weak` 还是 `unowned` Apple 建议,在访问时能够确定对象不会被释放的情况下优先使用 `unowned`,而在对象可能已经被释放的情况下则使用 `weak`。例如,在闭包中捕获 `self` 时,如果可以确保闭包在执行时 `self` 仍然存在,则使用 `unowned`;否则,使用 `weak`。 在实际开发中,需要根据具体的使用场景来选择合适的关键字。对于大多数情况,尤其是涉及到异步操作可变生命周期的对象时,推荐使用 `weak`,以确保程序的稳定性安全性[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

huhansome

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值