Objective-C

空指针、野指针、僵尸对象、内存泄露

2018-03-07  本文已影响27人  MichaelLedger

不管是多个对象还是单个对象,只要我们研究它的内存管理,就两点:
1.僵尸对象(野指针)
2.内存泄露

默认情况下, Xcode不会去检测指针指向的对象是否为一个僵尸对象(因为一旦开启,每次通过指针访问对象的时候.都会去检查指针指向的对象是否为僵尸对象,这样的话就影响效率了)。能访问就访问,不能访问就报错。
开启Xcode的僵尸对象检测,那么就会在通过指针访问对象的时候,检测这个对象是否为1个僵尸对象。如果是僵尸对象,就会报错。


自定义Student类,在main函数中添加下列代码

#import <Foundation/Foundation.h>
#import "HMStudent.h"
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        HMStudent *stu = [[HMStudent alloc]init];
        [stu setAge:18];
        [stu release];
        [stu setAge:20];
    }
    return 0;
}

运行程序,你会发现运行到[stu setAge:20]这句代码会报错,是个野指针错误:

1> 执行HMStudent *stu = [[HMStudent alloc]init];
内存中有个指针变量stu,指向了HMStudent实例对象

假设HMStudent实例对象的地址为0x10010,指针变量stu的内存地址为0xhhh045 ,则stu中存储的是HMStudent对象的地址0x10010。即指针变量stu指向了这个HMStudent对象。

2> 执行[stu setAge:18];
给stu所指向的HMStudent对象发送一条setAge:消息,即调用这个HMStudent对象的setAge:方法。目前来说,这个HMStudent对象仍存在于内存中,所以这句代码没有任何问题.

3> 执行[stu release];
给stu指向的HMStudent对象发送一条release消息。在这里,HMStudent对象接收到release消息后,会马上被销毁,所占用的内存会被回收。

HMStudent对象被销毁了,地址为0x10010的内存就变成了"垃圾内存",然而,指针变量stu仍然指向这一块内存,这时候,stu就称为了野指针!

4> 执行[stu setAge:20];
给stu所指向的HMStudent对象发送一条setAge:消息。但是HMStudent对象已经被销毁了,它所占用的内存已经是垃圾内存,如果你还去访问这一块内存,那就会报野指针错误。这块内存已经不可用了,也不属于你了,你还去访问它,肯定是不合法的。所以,这行代码报错了!

#import <Foundation/Foundation.h>
#import "HMStudent.h"
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        HMStudent *stu = [[HMStudent alloc]init];
        [stu setAge:18];
        stu = nil;
        [stu setAge:20];
    }
    return 0;
}

那么这个时候,stu变成了空指针,stu就不再指向任何内存了。
因为stu是个空指针,没有指向任何对象,因此[stu setAge:20]消息是发不出去的,不会造成任何影响。当然,肯定也不会报错。

  1. 利用野指针发消息是很危险的,会报错。也就是说,如果一个对象已经被回收了,就不要再去操作它,不要再尝试给它发消息。
  2. 利用空指针发消息是没有任何问题的,也就是说[nil setAge:20];是没有错误的。
  1. 僵尸对象调用方法,会报错,访问成员变量有时是可以的(但是这个是未知且不安全的)
  2. 为了防止不小心调用了僵尸对象,在对象被销毁之后, 将指向对象的指针变为空指针
  1. retain和release不匹配,retain多余release导致的内存泄露;
  2. 对象使用过程中,没有被release,而被赋值为nil;
  3. 在方法中不当的使用了retain;

参考:
OC野指针和空指针,僵尸对象和内存泄露
野指针与僵尸对象

上一篇 下一篇

猜你喜欢

热点阅读