OC学习5——内存管理(上)

36 篇文章 0 订阅

@一、范围任何继承了NSObject的对象,对基本数据类型无效。

@二、原理

1.每个对象内部都保存一个与之相关联的整数,称为引用计数器

2.当使用allocnew或者copy创建一个对象的时候,对象引用计数器被设置为1

3.给对象发送一条retain消息,可以使引用计数器值+1

4.给对象发送一条release消息,可以使引用计数器值-1

5.当一个对象的引用计数器值为0时,那么它将被销毁,其占用的内存被系统回收,OC也会自动向对象发送一条dealloc消息。一般会重写dealloc方法,在这里释放相关资源。一定不要直接调用dealloc方法。

6.可以给对象发送retainCount消息活得当前的引用计数器值。

@三、管理规则

1.谁创建,谁释放(谁污染,谁治理)。如果你通过allocnew或(mutablecopy来创建一个对象的话,那么你必须调用release或者autorelease。换句话说,不是你创建的你就不用释放。

2.一般来说,除了alloc,new,或者copy之外的方法创建的对象都被声明了autorelease

3.谁retain,谁release。只要你调用了retain,无论这个对象啊是如何生成的,你都要调用release

@四、示例

@interface Student : NSObject

@property int age;
@end
上面是Student类的声明部分.h文件,下面是实现部分.m文件

@implementation Student
@synthesize age = _age; // 在xcode4.5环境下可以省略

- (void)dealloc {
    NSLog(@"%@被销毁了", self);
    
   [super dealloc];
    // 一定要调用super的dealloc方法,而且最好放在最后面调用
}
@end
我们写一个测试方法来说明内存引用计数的变化

Student *stu = [[Student alloc] init]; // 1      // z代表无符号
    NSLog(@"count:%zi", [stu retainCount]);   
    [stu retain]; // 2  
    NSLog(@"count:%zi", [stu retainCount]);
    [stu release]; // 1  
    NSLog(@"count:%zi", [stu retainCount]);
    [stu release]; // 0  
    // [stu release]; // 会发生野指针错误,也就是说访问了不属于你的内存
再写一个示例说明:

// Student对象的计数器永远为1,所以不可能被释放
    [[Student alloc] init].age = 10;
    [Student new].age = 10;
    // 上面的代码都有内存泄露

@五、对象之间的内存管理

对象之间的内存管理是经常见到的情况,我们来模拟一下学生(Student)和书(Book)之间的内存管理。首先Student类声明和实现部分如下:

@interface Student : NSObject {
    Book *_book;
}
@property int age;

- (id)initWithAge:(int)age;
@property Book *book;

- (void)readBook;
@implementation Student

#pragma mark - 生命周期方法
#pragma mark 构造方法
- (id)initWithAge:(int)age {
    if ( self = [super init] ) {
        _age = age;
    }
    return self;
}
#pragma mark 回收对象
- (void)dealloc {
    // 释放Book对象
    [_book release]; 
    // [self.book release];
    NSLog(@"student:%i 被销毁了", _age);
    [super dealloc];
}
#pragma mark - getter和setter
// @synthesize book = _book;
// 如果自己手动实现了getter和setter,xcode就不会自动生成@synthesize
// 也就不会自动生成_book
// getter和setter的默认实现
- (void)setBook:(Book *)book {
    if (_book != book) {
        // 先释放旧的成员变量
        [_book release];
        // 再retain新传进来的对象
        _book = [book retain];
    }
}
- (Book *)book {
    return _book;
}
#pragma mark - 公共方法
#pragma mark 读书
- (void)readBook {
    NSLog(@"当前读的书是:%f", _book.price);
}
//#pragma mark - 私有方法
//#pragma mark 私有方法1
//- (void)test1 {}
//#pragma mark 私有方法2
//- (void)test2 {}
//#pragma mark 私有方法3
//- (void)test3 {}
@end
Book类的声明和实现部分如下:
@interface Book : NSObject
@property float price; // 价格

- (id)initWithPrice:(float)price;
@end
@implementation Book

- (id)initWithPrice:(float)price {
    if ( self = [super init] ) {
        _price = price;
    }
    return self;
}

- (void)dealloc {
    NSLog(@"book:%f 被销毁了", _price);
    
    [super dealloc];
}
@end

下面在main方法中对内存管理进行测试分析,示例如下:

void test(Student *stu) {
    // book:1
    Book *book = [[Book alloc] initWithPrice:3.5];

    // book:2
    stu.book = book;

    // book:1
    [book release];
    
    // book:1
    stu.book = book;
    stu.book = book;
    
     // book2:1
    Book *book2 = [[Book alloc] initWithPrice:4.5];
     // book2:2
    stu.book = book2;
    // book2:2
    stu.book = book2;
    // book2:1
    [book2 release];
    
    // book2:1
    stu.book = book2;
}

void test1(Student *stu) {
    [stu readBook];
}

int main(int argc, const char * argv[])
{

    @autoreleasepool {
        // stu:1
        Student *stu = [[Student alloc] initWithAge:10];
        
        // stu:1
        // book:1
        // book2:1
        test(stu);
        
        // stu:1
        // book:1
        // book2:1
        test1(stu);
        
        // stu:0
        // book2:0
        // book:1
        [stu release];
        
        // stu = nil;  清空stu这个指针,stu就会变成空指针
        
        // [stu release]; // 野指针(会报错)
        
        [nil release]; // 空指针(不会报错)
    }
    return 0;
}
通过分析,加深对内存管理的了解。熟记口诀,牢记原则。









  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值