小生第二次回顾这本书,上次还是两个月前才看完,记录于此,望兄去疑。
1、苹果和GNUstep内存管理区别
两者都通过引用计数来管理内存,但存储引用计数的方式有所不同。
苹果:大概通过散列表(引用计数表)来管理引用计数,表键值为内存块地址的散列值 引用计数---内存块地址。引用计数表各个记录中存有内存块,可以通过每条记录找到个对象的内存块。
GNUstep:引用计数直接保存在对象所占用内存块头部的变量中。```
2、藏着的autorelease和__strong
通过alloc、new、copy、mutableCopy开头的方法生成的对象只有自己持有,像咱们平时arrayWith、stringWith、[NSMutableDictionary dictionary]、等类方法生成的对象是不能持有该对象如
id object = [NSMutableArray array];
object是不能持有NSMutableArray生成的对象,但是为啥咱们在开发的时候还是可以正常使用object这个对象呢,其实系统默认给这个对象又加了一个方法 [object autorelease]方法,同时object对象默认用__strong修饰,强引用非自己生成的对象,通过该方法使取得对象存在。当object对象超出最近的NSAutoreleasePool作用域时,object对象被release释放,同时NSMutableArray对象也会被释放。
莽夫今天忘带电脑回家家,全用手机敲打。
3、__strong、__weak、__unsafe_unretained、__autoreleasing区别
id和对象类型在没有明确指定所有权修饰时,默认为__strong修饰符
id object = [[NSObject alloc] init];
<==> id __strong object = [[NSObject alloc] init];表示object对象对NSObject这个类对象的强引用。当两个或者多个__strong修饰的对象相互引用时会导致循环引用,内存不能回收造成内存泄漏、程序可能导致崩溃。
对自身的强引用 :id object = [[Text alloc] init]; [object setObject:object];
类成员变量循环引用:id obj0 = [[Text alloc] init]; id obj1 = [[Text alloc] init];
[obj0 setObject:obj1]; [obj1 setObject:obj0];
__weak和__unsafe_unretained都能解决循环引用的问题。不同之处在于如下图。。。
在ARC有效时用 @autoreleasepool快代替NSAutoreleasepool类,用附有__autoreleasing修饰变量来代替autorelease方法。
当有返回值时,由于return使得对象变量超出其作用域,编译器会自动将其注册到autoreleasepool中。
在访问附有__weak修饰的变量时,实际上必定要访问到注册到autoreleasepool的对象,__weak修饰变量只对对象持弱引用,在访问引用对象的过程中,该对象有可能被废弃,如果把要访问的对象注册到autoreleasepool中,那么在autoreleasepool块结束之前都能确保该对象存在。