zoukankan      html  css  js  c++  java
  • swift内存管理中的引用计数

    在swift中,每一个对象都有生命周期,当生命周期结束会调用deinit()函数进行释放内存空间。

    观察这一段代码:

    class Person{
        
        var name: String
        var pet: Pet?
        
        init(name: String){
            self.name = name
            print("Person", name, "is initialized")
        }
        
        init(name: String, petName: String){
            self.name = name
            self.pet = Pet(name: petName)
            print("Person", name, "is initialized")
        }
        
        deinit{
            print("Person", name, "is deinitialized!")
        }
    }
    
    class Pet{
        
        var name: String
        init(name: String){
            self.name = name;
            print("Pet", name, "is initialized")
        }
        
        deinit{
            print("Pet", name, "is deinitialized!")
        }
    }

    这段代码创建了两个类,分别是Person类和Pet类,每个类中都有init方法进行创建对象和deinit方法来释放内存空间,其中Person类中有两个init方法,分别对应着是否包含Pet类的名称。

    当我们调用这两个方法:

    var snow: Person? = Person(name: "snow", petName: "wolf")
    snow = nil

    两步的执行结果是:

    Pet wolf is initialized
    Person snow is initialized
    Person snow is deinitialized!
    Pet wolf is deinitialized!

    会发现在创建snow这个对象的时候调用的是第二个init方法,在这个方法中会创建一个新的Pet对象,因此会首先打印出Pet wolf is initialized然后是Person snow is initialized。当对snow对象进行内存释放的时候,将nil赋给这个对象,那么会释放snow这个内存空间,同时也会释放wolf这个内存空间。

    但是如果我们调用第一种init方法的时候我们会发现:

    var snow: Person? = Person(name: "snow")
    
    var wolf: Pet? = Pet(name: "wolf")
    snow?.pet = wolf
    
    snow = nil
    wolf = nil

    我们首先创建了一个snow对象,之后又创建了一个wolf对象,然后将wolf添加到snow对象中去,但是当我们对这snow这个对象进行内存释放的时候会发现:

    Person snow is initialized
    Pet wolf is initialized
    Person snow is deinitialized!

    仅仅只有snow的内存空间被释放了,但是wolf的内存空间并没有被释放,这里就和swift内存管理中的引用计数有关了:

    当我们创建了snow这个对象之后,我们就为它开辟了一个内存空间,命名为a,这时候snow这个对象引用了这片内存空间,这片内存空间的引用计数就是1,

    同样地当我们创建了wolf这个对象之后,我们就为它开辟了一个内存空间,命名为b,这时候wolf这个对象引用了这片内存空间,这片内存空间的引用计数就是1,

    当我们将snow?.pet = wolf之后,那么snow中的一个属性也指向了创建wolf这个对象的内存空间,那么这篇内存空间的引用计数就是2.

    当我们对snow = nil进行内存空间的释放,那么内存空间a的引用计数就为0了,同时内存空间b的引用计数就为1了。

    当系统发现一篇内存空间的引用计数为0,那么,系统就会释放这片内存空间,此时内存空间a就被释放了。

    但是内存空间b的引用计数为1,系统不会进行自动的内存释放。只有当我们进行:

    wolf = nil

    操作之后,这片内存空间b才会被释放。

    同样地对于这样代码:

    import UIKit
    
    class Person{
        
        var name: String
        
        init(name: String){
            self.name = name
            print("Person", name, "is initialized")
        }
        
        deinit{
            print("Person", name, "is being deinitialized!")
        }
    }
    
    var person1: Person? = Person(name: "liuyubobobo")
    var person2: Person? = person1
    var person3: Person? = person1

    那么person1的内存空间的引用计数为3,如果释放这片内存空间的话,需要将三个对象都为nil

    如果仅仅是将person1=nil的话,并不会释放这一片内存空间。

  • 相关阅读:
    转:sql语句中GROUP BY 和 HAVING和使用 count()
    shell中的大括号和小括号
    转:关于rename命令ubuntu下的用法
    Linux批量重命名
    STL 源代码剖析 算法 stl_algo.h -- partition
    HDU 5091 线段树扫描线
    IBM 中国研究院面试经历
    当人手一部智能手机时 庞大的数据中心们已死
    Treap的读书笔记2
    【JUnit4.10源码分析】5 Statement
  • 原文地址:https://www.cnblogs.com/winterfells/p/7476485.html
Copyright © 2011-2022 走看看