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引用计数的内部实现是啥样的?
可以看到在实例t3内存里面保存有实例的引用计数,但是需要右移33位才能获取到真正的引用计数,具体原理我们可以看下源码是怎么实现;
image
可以看到在结构体
struct HeapObject
里面有一个变量refCounts
,这个变量实际上就是用来表示引用计数相关信息的,我们继续看源码,最终可以看到结构体成员变量BitsType代表的refCounts就是一个uint64_t
类型的数据。image
再往下找,我们可以找到RefCountBitsT的初始化方法,实现会将引用计数相关的信息通过左移保存在各个bit位上,下面总结了一份各个位数代表的含义
image
下图是每一份代表的含义,可以33-62位就是代码swift里面的实例的引用计数;所以我们获取到引用计数的地址时,一般需要右移33位才能得到真正的引用计数
image
网友评论