Q.1:引用計數到底是在指針上還是在實際對象內存中
Q.2:Effective Objc里面提到的被strong修飾的屬性在設置新值得時候會保留新值釋放舊值是什么意思
Q.3:調用set方法賦值和不調用set方法賦值對引用計數有什么影響
Q.4:為什么有時候引用計數為0后,依然可以正常調用該對象的變量和方法
由于ARC不允許直接調用retainCount和retain,release方法所以使用運行時的庫進行研究
#import <objc/runtime.h>
//.h文件中定義屬性
@property(nonatomic,strong) objectA *a;
Q.1比較好解釋,在.m文件中不對a初始化查看引用計數會崩潰,而初始化之后就不會了,可以很明顯的得出引用計數是針對指針所指向的對象的而不是指針本身
Q.2
NSLog(@"%ld", CFGetRetainCount((CFTypeRef)_a));
//.m文件中定義
NSObject *b = [[objectA alloc] init];
CFRetain((__bridge CFTypeRef)(b));
CFRetain((__bridge CFTypeRef)(b));
CFRetain((__bridge CFTypeRef)(b));//b:RefCt = 4
NSObject *c = [[objectA alloc] init];//c:RefCt = 1
_a = c;//先不調用set方法避免干擾
/*
以上代碼輸出后引用計數輸出為
a:RefCt = 2,b:RecCt = 2
a和b引用計數均+1
*/
_a = b;
/*
以上代碼輸出后引用計數輸出為
a:RefCt = 5,b:RecCt = 5,c:RefCt = 1
a引用計數變5,b引用計數+1,c引用計數-1
*/
綜上:可以得出strong關鍵字修飾的屬性在設置新值的時候流程是先把將要賦給該指針的值引用計數+1,然后讓改指針原本所指向的內存引用計數-1,又因為引用計數是針對實際對象的而不是針對指針的Q.1的結論,故將該指針指向新值指針所在的內存的時候,引用計數直接變成新值得引用計數,
的保留新值和釋放舊值不如說翻譯成[新值 retain],[舊值 release]比較好
Q3.這個問題我也還沒有具體搞清楚,只知道使用self.object打印引用計數的時候會比_obejct多1,但是打印過后引用計數又會恢復原樣,可能是self多引用了一次?
Q.4我糾結了好久,在MRC研究引用計數的的時候經常出現這種情況,不管是release之后引用計數變0還是直接dealloc掉一個對象都有能繼續正常操作該對象的現象
最后答案其實是
如果調用release后,基于某些原因,引用計數降至0,那么number對象所占內存也許會被回收,這樣的話再調用NSLog可能會使程序崩潰,筆者在這里只說可能,而沒說一定,因為對象所占的內存在‘接觸分配’后,只是返回可用內存池,如果執行log時尚未復寫對象內存,那么該對象依然有效,這是程序不會崩潰
該答案來自effective objective-c 2.0