swift 内存管理之类的引用计数

swift使用自动引用计数(ARC)来管理内存

默认情况下swift的类初始化的时候,引用计数默认就设置为1;我们可以使用函数public func CFGetRetainCount(_ cf: CFTypeRef!) -> CFIndex来打印类的实例对象的引用计数,但需要注意的是,使用CFGetRetainCount的时候会把要打印对象的引用计数先加1,也就是说,打印出来的引用计数会多1;见如下代码

class JFTest{
    var age:Int?
    var name:String?
}

let t1 = JFTest()
print(CFGetRetainCount(t1))
let t2 = t1
let t3 = t1
print(CFGetRetainCount(t1))

输出的结果如下:


image

使用LLDB查看实例的内存

那swift的引用计数在内存里面是怎么保存的了?我们可以使用LLDB打印一下,看一下swift引用计数的内部实现是啥样的?

image

可以看到在实例t3内存里面保存有实例的引用计数,但是需要右移33位才能获取到真正的引用计数,具体原理我们可以看下源码是怎么实现;
image

可以看到在结构体struct HeapObject里面有一个变量refCounts,这个变量实际上就是用来表示引用计数相关信息的,我们继续看源码,最终可以看到结构体成员变量BitsType代表的refCounts就是一个uint64_t类型的数据。
image

再往下找,我们可以找到RefCountBitsT的初始化方法,实现会将引用计数相关的信息通过左移保存在各个bit位上,下面总结了一份各个位数代表的含义
image

下图是每一份代表的含义,可以33-62位就是代码swift里面的实例的引用计数;所以我们获取到引用计数的地址时,一般需要右移33位才能得到真正的引用计数
image

你可能感兴趣的:(swift 内存管理之类的引用计数)