斯威夫特有:

强引用 弱引用 无主的引用

无主引用与弱引用有何不同?

什么时候使用无主引用是安全的?

无主引用是否像C/ c++中的悬浮指针一样存在安全风险?


当前回答

弱引用和无主引用都不会在被引用的对象上创建强持有(也就是说,它们不会增加保留计数以防止ARC释放被引用的对象)。

但为什么只有两个关键词呢?这种区别与可选类型是Swift语言内置的事实有关。长话短说:可选类型提供了内存安全(这与Swift的构造函数规则非常配合——为了提供这种好处,构造函数规则非常严格)。

弱引用允许它变成nil(当被引用的对象被释放时自动发生),因此你的属性类型必须是可选的——所以作为程序员,你有义务在使用它之前检查它(基本上编译器会强迫你尽可能多地编写安全代码)。

无主引用假定它在生命周期内永远不会变为nil。在初始化过程中必须设置无主引用——这意味着该引用将被定义为非可选类型,无需检查即可安全使用。如果被引用的对象以某种方式被释放,那么当使用无主引用时,应用程序将崩溃。

苹果文档中写道:

只要该引用成为有效,就使用弱引用 在生命周期的某个时刻为Nil。相反,使用无主的 当你知道引用永远不会为nil时,再引用 已在初始化时设置。

在文档中,有一些例子讨论了保留周期以及如何打破它们。所有这些例子都是从文档中摘录出来的。

关键字weak的示例:

class Person {
    let name: String
    init(name: String) { self.name = name }
    var apartment: Apartment?
}
 
class Apartment {
    let number: Int
    init(number: Int) { self.number = number }
    weak var tenant: Person?
}

现在,对于一些ASCII艺术(你应该去看看文档-他们有漂亮的图表):

Person ===(strong)==> Apartment
Person <==(weak)===== Apartment

Person和Apartment示例显示了一种情况,其中两个属性都被允许为nil,有可能导致强引用循环。这种情况最好使用弱引用来解决。两个实体可以不严格依赖于另一个实体而存在。

u主关键字示例:

class Customer {
    let name: String
    var card: CreditCard?
    init(name: String) { self.name = name }
}
 
class CreditCard {
    let number: UInt64
    unowned let customer: Customer
    init(number: UInt64, customer: Customer) { self.number = number; self.customer = customer }
}

在本例中,客户可能拥有也可能没有信用卡,但信用卡始终与客户相关联。为了表示这一点,Customer类有一个可选的card属性,而CreditCard类有一个非可选的(无主的)Customer属性。

Customer ===(strong)==> CreditCard
Customer <==(unowned)== CreditCard

Customer和CreditCard示例展示了这样一种情况:允许一个属性为nil,而另一个属性不能为nil,这可能会导致强引用循环。这种情况最好使用无主引用来解决。

苹果公司的说明:

弱引用必须声明为变量,以表明它们的 值可以在运行时更改。弱引用不能声明为 常数。

还有第三种情况,两个属性都应该有一个值,并且在初始化完成后,两个属性都不应该为nil。

在使用闭包时,还有一些经典的保留周期场景需要避免。

为此,我鼓励你访问苹果文档,或阅读这本书。

其他回答

Q1。“无主引用”与“弱引用”有何不同?

弱引用:

弱引用是指不能保持强引用的引用 它引用的实例,因此不会阻止ARC处理 引用的实例。因为弱引用是允许的 " no value "时,必须声明每个弱引用都具有 可选的类型。(苹果文档)

无主参考:

与弱引用一样,无主引用不会保持强持有 在它引用的实例上。然而,与弱引用不同的是 无主引用假定总是有一个值。正因为如此, 无主引用总是定义为非可选类型。(苹果文档)

何时使用:

只要该引用成为有效,就使用弱引用 在生命周期的某个时刻为Nil。相反,使用无主的 当你知道引用永远不会为nil时,再引用 已在初始化时设置。(苹果文档)


Q2。什么时候使用“无主引用”是安全的?

如上所述,假定无主引用总是有一个值。所以你应该只在确定引用永远不会为nil时使用它。Apple Docs通过以下示例演示了无主引用的用例。

假设我们有两个类Customer和CreditCard。客户可以没有信用卡而存在,但信用卡不会没有客户而存在,也就是说,可以假设信用卡总是有客户。所以,它们应该有这样的关系:

class Customer {
    var card: CreditCard?
}

class CreditCard {
    unowned let customer: Customer
}

第三季。在C/ c++中,“无主引用”引用是否像“悬空指针”一样存在安全风险

我不这么想。

由于无主引用只是保证有值的弱引用,因此无论如何都不应构成安全风险。然而,如果你试图访问一个无主引用后,它引用的实例被释放,你将触发一个运行时错误,应用程序将崩溃。

这是我看到的唯一风险。

链接到Apple Docs

如果self可以在闭包中为nil,请使用[weak self]。

如果self在闭包中永远不会为nil,请使用[无主的self]。

如果当你使用[un主self]时它崩溃了,那么self在闭包中的某个时候可能是nil,你可能需要使用[weak self]来代替。

查看在闭包中使用strong、weak和u主的示例:

https://developer.apple.com/library/ios/documentation/swift/conceptual/swift_programming_language/AutomaticReferenceCounting.html

弱引用和无主引用都不会在被引用的对象上创建强持有(也就是说,它们不会增加保留计数以防止ARC释放被引用的对象)。

但为什么只有两个关键词呢?这种区别与可选类型是Swift语言内置的事实有关。长话短说:可选类型提供了内存安全(这与Swift的构造函数规则非常配合——为了提供这种好处,构造函数规则非常严格)。

弱引用允许它变成nil(当被引用的对象被释放时自动发生),因此你的属性类型必须是可选的——所以作为程序员,你有义务在使用它之前检查它(基本上编译器会强迫你尽可能多地编写安全代码)。

无主引用假定它在生命周期内永远不会变为nil。在初始化过程中必须设置无主引用——这意味着该引用将被定义为非可选类型,无需检查即可安全使用。如果被引用的对象以某种方式被释放,那么当使用无主引用时,应用程序将崩溃。

苹果文档中写道:

只要该引用成为有效,就使用弱引用 在生命周期的某个时刻为Nil。相反,使用无主的 当你知道引用永远不会为nil时,再引用 已在初始化时设置。

在文档中,有一些例子讨论了保留周期以及如何打破它们。所有这些例子都是从文档中摘录出来的。

关键字weak的示例:

class Person {
    let name: String
    init(name: String) { self.name = name }
    var apartment: Apartment?
}
 
class Apartment {
    let number: Int
    init(number: Int) { self.number = number }
    weak var tenant: Person?
}

现在,对于一些ASCII艺术(你应该去看看文档-他们有漂亮的图表):

Person ===(strong)==> Apartment
Person <==(weak)===== Apartment

Person和Apartment示例显示了一种情况,其中两个属性都被允许为nil,有可能导致强引用循环。这种情况最好使用弱引用来解决。两个实体可以不严格依赖于另一个实体而存在。

u主关键字示例:

class Customer {
    let name: String
    var card: CreditCard?
    init(name: String) { self.name = name }
}
 
class CreditCard {
    let number: UInt64
    unowned let customer: Customer
    init(number: UInt64, customer: Customer) { self.number = number; self.customer = customer }
}

在本例中,客户可能拥有也可能没有信用卡,但信用卡始终与客户相关联。为了表示这一点,Customer类有一个可选的card属性,而CreditCard类有一个非可选的(无主的)Customer属性。

Customer ===(strong)==> CreditCard
Customer <==(unowned)== CreditCard

Customer和CreditCard示例展示了这样一种情况:允许一个属性为nil,而另一个属性不能为nil,这可能会导致强引用循环。这种情况最好使用无主引用来解决。

苹果公司的说明:

弱引用必须声明为变量,以表明它们的 值可以在运行时更改。弱引用不能声明为 常数。

还有第三种情况,两个属性都应该有一个值,并且在初始化完成后,两个属性都不应该为nil。

在使用闭包时,还有一些经典的保留周期场景需要避免。

为此,我鼓励你访问苹果文档,或阅读这本书。

当你确定在你访问self时,self永远不会为nil时,使用u主。

示例(当然,你可以直接从MyViewController中添加目标,但同样,这是一个简单的例子):

class MyViewController: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()

        let myButton = MyButton { [unowned self] in
            print("At this point, self can NEVER be nil. You are safe to use unowned.")
            print("This is because myButton can not be referenced without/outside this instance (myViewController)")
        }
    }
}

class MyButton: UIButton {
    var clicked: (() -> ())

    init(clicked: (() -> ())) {
        self.clicked = clicked

        // We use constraints to layout the view. We don't explicitly set the frame.
        super.init(frame: .zero)

        addTarget(self, action: #selector(clicked), for: .touchUpInside)
    }

    @objc private func sendClosure() {
        clicked()
    }
}

当你访问self时,有可能self值为nil时,使用weak。

例子:

class MyViewController: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()

        NetworkManager.sharedInstance.receivedData = { [weak self] (data) in
            print("Can you guarentee that self is always available when the network manager received data?")
            print("Nope, you can't. Network manager will be alive, regardless of this particular instance of MyViewController")
            print("You should use weak self here, since you are not sure if this instance is still alive for every")
            print("future callback of network manager")
        }
    }
}

class NetworkManager {

    static let sharedInstance = NetworkManager()

    var receivedData: ((Data) -> ())?

    private func process(_ data: Data) {
        // process the data...

        // ... eventually notify a possible listener.
        receivedData?(data)
    }
}

无主的缺点:

效率比弱 你可以(好吧,你是被迫的)将实例标记为不可变(从Swift 5.0开始就不再这样了)。 向代码的读者表明:这个实例与X有关系,没有它就不能生存,但是如果X消失了,我也就消失了。

弱者的缺点:

比无主更安全(因为它不会崩溃)。 可以创建与X的双向关系,但两者都可以没有对方。

如果你不确定,就用weak。等等,我的意思是在StackOverflow上问问你在这种情况下应该怎么做!在不应该使用weak的情况下一直使用weak只会让您和代码的读者感到困惑。

无主引用是一种弱引用,用于两个对象之间存在同一生命周期关系的情况下,这时一个对象应该只属于另一个对象。它是一种在对象与其属性之一之间创建不可变绑定的方法。

In the example given in the intermediate swift WWDC video, a person owns a credit card, and a credit card can only have one holder. On the credit card, the person should not be an optional property, because you don't want to have a credit card floating around with only one owner. You could break this cycle by making the holder property on the credit a weak reference, but that also requires you to make it optional as well as variable (as opposed to constant). The unowned reference in this case means that although CreditCard does not have an owning stake in a Person, its life depends on it.

class Person {
    var card: CreditCard?
}

class CreditCard {

    unowned let holder: Person

    init (holder: Person) {
        self.holder = holder
    }
}