iOS:关于修饰符的那些事
关于iOS修饰符
首先强调一下:
强引用:当前对象被其他对象引用时,会执行retain操作,引用计数器+1。当retainCount=0时,该对象才会被销毁。因为我们要进行对象的内存管理,所以这是默认的引用方式。(默认是强引用)
弱引用:当前对象的生命周期不被是否由其他对象引用限制,它本该什么时候销毁就什么时候被销毁。即使它的引用没断,但是当它的生存周期到了时就会被销毁。
所以一般来说,类“内部”的属性设置为strong,类“外部”的属性设置为weak。说到底就是一个归属权的问题。小心出现循环引用导致内存无法释放。
1.NSNumber 是不可变的( immutable )
使用 Copy 是没有意义的,系统不会给它分配新的内存空间,应该使用Strong修饰
2.NSString用Copy还是Strong
给NSString属性赋值,如果是不可变对象,Strong和Copy 引用计数都会+1
如果是可变对象,copy会深copy对象,Strong的引用计数+1,Copy的引用计数此时+2;
当源字符串是NSString时,由于字符串是不可变的,所以,不管是strong还是copy属性的对象,都是指向源对象,copy操作只是做了次浅拷贝。
当源字符串是NSMutableString时,strong属性只是增加了源字符串的引用计数,而copy属性则是对源字符串做了次深拷贝,产生一个新的对象,且copy属性对象指向这个新的对象。
但是改属性的类型始终都是NSString,而不是NSMutableString。
一般我们将对象声明为NSString时,都不希望它改变,所以大多数情况下,我们建议用copy。
3.用字面量语法给NSString赋值
使用字面量语法赋值了,类似于
NSString*str1=@"1234567890";
这样的方法创建的字符串,retainCount作为NSInterger打印为-1,其实最其值应为4294967295,这意味着无限的“retainCount”,也就是说这个对象是不能被释放的。
所以用这样的方式创建字符串,无论创建多少次,他们都指向同一个对象(同样的内存地址)。
4.NSString被不同修饰符操作
对一个NSString进行retain和copy操作,最终仍然是自己,引用计数+1;但是进行MutableCopy会将其Copy到堆上,操作完成后引用计数为1。
5.weak和strong的区别
两者不同的是 当一个对象不再有strong类型的指针指向它的时候 它会被释放 ,即使还有weak型指针指向它。
只要最后一个strong型指针不再指向对象,那么对象就会被释放,同时所有的weak型指针都将会被清除。
6.assgin和weak的区别
weak表示了一种(non-relationship)非拥有关系,为这类型属性赋值时,set方法既不保留新值,也不会释放旧值。
assign也可以用来修饰对象,但是使用assign修饰的对象在释放后,指针还是存在的,也就是说指针没有置为nil,会造成野指针。assign修饰的基本数据类型(例如NSInteger,CGFloat等)和C数据类型(int,float,double,char)等一般分配在栈空间上,栈空间的内存会由系统自动处理,当分配的栈空间的内存没有被指针指向时就会被销毁,所以不会造成野指针异常。
weak比assign多了一个功能就是在当前属性消失(引用计数为0)后,会自动赋值为nil;这样给weak修饰的属性发消息就不会因为野指针造成crash。
此外,还有一个修饰符unsafe_unretained,这个修饰符跟weak类似,区别是被指向对象消失后,属性不会自动赋值为nil。
7.NSString的Retain和Copy的区别
Copy其实是建立了一个相同的对象,而retain不是:
比如一个NSString对象,地址为0×1111,内容为@”STRING”
Copy到另外一个NSString之 后,地址为0×2222,内容相同,新的对象retain为1, 旧有对象没有变化
retain到另外一个NSString之 后,地址相同(建立一个指针,指针拷贝),内容当然相同,这个对象的retain值+1
简单的说,二者的区别就是,Copy是内存Copy,而retain是指针拷贝。
8.Weak指针置为nil的实现原理
Weak指针不会增加所引用对象的计数,并在引用对象被回收的时候自动被置为nil,这个通常用来解决循环引用的问题。
对象被释放的时候, 其dealloc方法执行之前, 它的所有weak属性都已经被设置为nil. 因此, 如果期望在dealloc里访问weak属性, 那是不行的.
内部实现 —— Weak表
Runtime维护了一个Weak表,用于存储指向某个对象的所有Weak指针。Weak表其实是一个哈希表,Key是所指对象的地址,Value是Weak指针的地址(这个地址的值是所指对象的地址)的数组。
在对象被回收的时候,经过层层调用,会最终触发下面的方法将所有Weak指针的值设为nil
简单来说,这个方法首先根据对象地址获取所以Weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从Weak表中删除。
http://www.cnblogs.com/HypeCheng/p/4685443.html
9.__block和__weak
http://www.cnblogs.com/yajunLi/p/6203222.html?utm_source=itdadao&utm_medium=referral
block下循环引用的问题
__block本身并不能避免循环引用,避免循环引用需要在block内部把__block修饰的obj置为nil
__weak可以避免循环引用,但是其会导致外部对象释放了之后,block 内部也访问不到这个对象的问题,我们可以通过在 block 内部声明一个 __strong
的变量来指向 weakObj,使外部对象既能在 block 内部保持住,又能避免循环引用的问题
__block与__weak功能上的区别。
__block会持有该对象,即使超出了该对象的作用域,该对象还是会存在的,直到block对象从堆上销毁;而__weak仅仅是将该对象赋值给weak对象,当该对象销毁时,weak对象将指向nil;
__block可以让block修改局部变量,而__weak不能。
另外,MRC中__block是不会引起retain;但在ARC中__block则会引起retain。所以ARC中应该使用__weak。
10.可变属性和不可变属性
当修饰可变属性(如NSMutableArray,NSMutableDictionary,NSMutableString)时,用Strong。
当修饰不可变属性时,则用Copy。