一、简介
深入了解ARC,主要从一下几个修饰符深入了解 __strong/__weak/_atuoreleasing
1、__strong
首先时当编译器初始化strong类型对象时,编译器伪代码如下
id obj = objc_msgSend(NSObject,@selector(alloc));
objc_msgSend(obj,@selector(init));
objc_release(obj);
生成对象时,是以消息发送的方式去分配内存空间和对象,后用release方法释放对象,当为ARC时不需显示调用release方法,编译器会插入该方法。
相对于系统的 alloc/new/copy/mutablecopy 来说
另一些方法初始化如:id obj=[NSMutableArray array];
id obj = objc_msgSend(NSObject,@selector(alloc));
objc_retainAutoreleasedReturnVlaue(obj); //编译器插入 用于优化程序,相对应的成对get方法时使用:objc_autoreleaseReturnValue(obj); 通过该过程的两个方法 在初始化和获取后调用,就不必将对象注册autoreleaspool中,而是直接返回该对象,优化了运行程序时间及步骤
objc_release(obj);
2、__weak修饰符
有weak 初始化的对象,是不能被持有,会被立即释放;编译器会生成一个全局的弱引用表;当对弱引用对象操作时不会做引用计数加减操作;
编译器释放弱引用对象是怎样的
1)从weak表中获取释放对象的地址为键值的记录
2)将记录的__weak修饰的变量地址,并赋值为nil;并从表中删除该记录
初始化对象时,不可直接 用__weak 否则会直接被释放
注意:如没循环引用建议不要插入weak,大量操作会影响cpu性能
3、__autorelease修饰符
用该修饰符初始化的对象,等同于在mrc 中,插入了autorelease;使得对象在不使用时能得以释放
知识点补充:
深拷贝浅拷贝
理解NSString使用copy及strong修饰的原理
1. copy出来的字符串一定是不可变字符串,如果传入的是可变字符串,会发生深拷贝为不可变字符串,否则为浅拷贝。
2. mutablecopy,一定是深拷贝,拷贝出来的一定是可变字符串或者数组,即使传入的是不可变字符串或者数组。
理解:
为什么NSString使用copy修饰也就可以理解了。使用copy修饰之后,即使属性拷贝来自可变字符串,也会被深拷贝成不可变字符串,也就是源字符串修改之后不会影响到属性字符串,增强了代码的健壮性。
关于不可变字符串和数组的copy是浅拷贝也很好理解,既然数据源本身是不可变的,也就是具备安全性,那么系统默认浅拷贝其中数据,显然是合理的做法。
Tagged Pointer技术,用于优化NSNumber、NSDate、NSString等小对象的存储,使用Tagged Pointer之后,NSNumber指针里面存储的数据变成了:Tag + Data,也就是将数据直接存储在了指针中那么他就不是一个对象了。
1 当字符串很短的时候iOS会使用Tagged Pointer对NSString进行优化,此时NSString内存地址是Tag + Data,那么它就不是一个对象了,因此它也就没有setName方法,所以他也就不会调用setName方法而是直接赋值
2 那么当字符串很长,那么就不会使用Tagged Pointer技术,那么他就是一个普通的字符串对象,当开启多个线程并行调用setName方法时在调用[_name release];时有可能_name已经被上一个线程给release过了,它的引用计数为零了,那么这是再有一个线程调用release就会报坏内存访问。解决办法就是name声明成atomic,或者在调用self.name时加锁。