随意细解:OC -- 内存管理

来源:互联网 发布:网络推广员有前途吗 编辑:程序博客网 时间:2024/05/10 12:55

内存管理的方式

垃圾回收(gc)

ios⽀持两种内存管理⽅式:ARC和MRC。而垃圾回收在电脑上还是有的。

MRC

Manual Reference Count,⼈⼯引⽤计数:内存的开辟和释放都由程序代码进⾏控制。相对垃圾回收来说,对内存的控制更加灵活,可以在需要释放的时候及时释放,对程序员的要求较⾼,程序员要熟悉内存管理的机制。

ARC

Auto Reference Count,⾃动引⽤计数:iOS 5.0的编译器特性,它允许⽤户只开辟空间,不⽤去释放空间。它不是垃圾回收!它的本质还是MRC,只是编译器帮程序员默认加了释放的代码。

内存管理机制

OC采⽤引⽤计数机制管理内存,当⼀个新的引⽤指向对象时,引⽤计数器就递增,当去掉⼀个引⽤时,引⽤计数就递减。当引⽤计数到零时,该对象就将释放占有的资源。

影响引⽤计数的⽅法

  1. alloc: 开辟空间,引用计数 从 0 变成 1

     Person *p1 = [[Person alloc]initWithName:@"aaa" Age:22]; NSLog(@"%ld", p1.retainCount);
  2. retain:使引用计数 + 1

    [p1 retain];
  3. release:影响引用计数 - 1

    当对象的引用计数为0的时候,该对象就会被系统释放。就不要去访问这个空间了,一访问就是野指针

    [p1 release];
  4. autorelease:需要自动释放池 去 - 1
    在未来的某一时刻会自动 - 1,是依托自动释放池去释放的。出了自动释放池 相当于系统会给释放池中间调用了autorelease对象发送一个release消息。

    Person *p1 = [[Person alloc]initWithName:@"aaa" Age:21];NSAutoreleasePool *pool = [[NSAutoreleasePool alloc]init];[p1 autorelease];NSLog(@"%ld", p1.retainCount);[pool release]; NSLog(@"%ld", p1.retainCount);

    或者:

    // 自动释放池@autoreleasepool {    [p1 autorelease];}

dealloc方法

-dealloc是继承⾃⽗类的⽅法,当对象引⽤计数为0的时候,由对象⾃动调⽤。可以在自定义类中重写dealloc方法。

书写规范:一定要写在类的最上面

- (void)dealloc{// 如果你要在dealloc中使用对象的话,一定要写在[super dealloc]的上面,避免出现野指针问题NSLog(@"释放了");// 如果重写dealloc方法 必须要调用父类的方法,保证对象可以被释放[super dealloc];}

内存管理的基本原则

引⽤计数的增加和减少相等,当引⽤计数降为0之后,不应该再使⽤这块内存空间。凡是使⽤了alloc、retain或者copy让内存的引⽤计数增加了,就需要使⽤release或者autorelease让内存的引⽤计数减少。在⼀段代码内,增加和减少的次数要相等。

copy

伪拷贝

拷贝完还是一个对象(直接返回)

- (id)copyWithZone:(NSZone *)zone{    return nil;}

浅拷贝

拷贝后 有2个对象,但是他们的值是同一个值。
引用计数变化:被拷贝的对象不变,拷贝出来的新对象从 0 - 1

- (id)copyWithZone:(NSZone *)zone{    // 浅拷贝     Person *p = [[Person alloc]initWithName:_name Age:_age];    return p;}

深拷贝

深拷贝:拷贝出新对象,并且对象的值也重新拷贝一份再赋值。
对字符串进行拷贝,拷贝的结果要看字符串这个类如何实现的拷贝方法,对不可变字符串的拷贝,其实相当于直接retain一次。可变字符串拷贝时就是真拷贝了一个新的出来。

- (id)copyWithZone:(NSZone *)zone{     NSString *name = [_name copy];     Person *p = [[Person alloc]initWithName:name Age:_age];    return p;}
0 0
原创粉丝点击