1.6.4 IOS 内存管理alloc,assign,retain,copy,release,autorelease

IOS的对象都继承于NSObject, 该对象有一个方法:retainCount ,内存引用计数。

 

alloc对象分配后引用计数为1
retain 对象的引用计数+1
copy 一个对象变成新的对象(新内存地址) 引用计数为1, 原来对象计数不变
release 对象引用计数-1 如果为0释放内存
autorelease 对象引用计数-1 如果为0不马上释放,最近一个pool时释放

存管理的原则是最终的引用计数要平衡:

    如果最终引用计数大于0,则会内存泄露
    如果当引用计数等于0时还对该对象进行操作,则会出现内存访问失败而crash ,所以尽量设置为nil
    这两个问题都很严重,所以请一定注意内存释放和不用过后设置为nil。

 

 

 

区别(assign,retain,copy):

 

NSString *str = [[NSString alloc] initWithString:@'abc'];

上面一段代码会执行以下两个动作:

 a. 在堆上分配一段内存用来存储@'abc' ,比如:内存地址为0X1111 内容为 'abc'。

 b. 在栈上分配一段内存用来存储str,比如:地址为0XAAAA 内容自然为0X1111。

 

1.assign的情况:NSString *newStr = [str assign];

newStr和str完全相同,地址都是0XAAAA ,内容为0X1111 ,即newStr只是str的别名,对任何一个操作就等于对另一个操作。因此retainCount不需要增加。相当于指针赋值,不对引用计数进行操作,注意原对象不用时,一定要把这个设置为nil。

2.retain的情况:NSString * newStr = [str retain];

newStr的地址不再为0XAAAA,可能为0XAABB,但是内容依然为0X1111.因此newStr和str都可以管理'abc'所在的内存。因此 retainCount需要增加1。相当于对原对象的引用计数加1

3.copy的情况:NSString * newStr = [str copy];

会在堆上重新开辟一段内存存放@'abc',比如0X1122,内容为@'abc,同时会在栈上为newStr分配空间,比如地址:0XAACC,内容为0X1122,因此retainCount增加1供newStr来管理0X1122这段内存.

 

 

 

 

区别(assign与retain):

1. 接触过C,那么假设你用malloc分配了一块内存,并且把它的地址赋值给了指针a,后来你希望指针b也共享这块内存,于是你又把a赋值给(assign)了b。此时a和b指向同一块内存,请问当a不再需要这块内存,能否直接释放它?答案是否定的,因为a并不知道b是否还在使用这块内存,如果a释放了,那么b在使用这块内存的时候会引起程序crash掉。
2. 了解到1中assign的问题,那么如何解决?最简单的一个方法就是使用引用计数(reference counting),还是上面的那个例子,我们给那块内存设一个引用计数,当内存被分配并且赋值给a时,引用计数是1。当把a赋值给b时引用计数增加到2。这时如果a不再使用这块内存,它只需要把引用计数减1,表明自己不再拥有这块内存。b不再使用这块内存时也把引用计数减1。当引用计数变为0的时候,代表该内存不再被任何指针所引用,系统可以把它直接释放掉。
总结:上面两点其实就是assign和retain的区别,assign就是直接赋值,从而可能引起1中的问题,当数据为基础数据类型(NSInteger,CGFloat)和C数据类型(int, float, double, char)时,可以使用assign。retain就如2中所述,使用了引用计数,retain引起引用计数加1, release引起引用计数减1,当引用计数为0时,dealloc函数被调用,内存被回收。

 

区别(atomic与nonatomic):

atomic、nonatomic,决定编译器生成的getter和setter是否为原子操作。

 

        atomic

                设置成员变量的@property属性时,默认为atomic,提供多线程安全。

                在多线程环境下,原子操作是必要的,否则有可能引起错误的结果。加了atomic,setter函数会变成下面这样:
                        {lock}
                                if (property != newValue) { 
                                        [property release]; 
                                        property = [newValue retain]; 
                                }
                        {unlock}

        nonatomic

        禁止多线程,变量保护,提高性能。

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值