iOS 内存管理(二)

一、简介

深入了解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时加锁。

你可能感兴趣的:(iOS 内存管理(二))