第06天OC语言(02):自动释放池注意事项

  • 不要等到明天,明天太遥远,今天就行动。
须读:看完该文章你能做什么?

知道对象什么时候才会加到autoreleasepool里面,
在autoreleasepool什么时候才释放对象,
如何处理比较占用内存的对象

学习前:你必须会什么?(在这里我已经默认你具备C语言的基础了)

什么是autoreleasepool


一、本章笔记
项目1
 1.一定要在自动释放池中 调用 autorelease,才会将对象方法自动释放池中
 2. 在自动释放池中创建了对象,移动要调用 autorelease,才会将对象放到自动释放池中
 3.只要在自动释放池中 调用 autorelease, 就会将对象放入自动释放池中
 4.一个程序中 可以创建N个自动释放池, 并且自动释放池 还可以嵌套
     如果存在多个自动释放池的时候,自动释放池 是以 "栈"的形式存储的
     栈的特点: 先进后出
项目2
 1.不要在自动释放池 使用比较消耗内存的对象,占用内存比较大的对象
  2.尽量不要在自动释放池使用循环, 特别是循环的次数非常多,并且非常占用内存

二、项目1
main.m
#pragma mark 02-自动释放池注意事项
#pragma mark 概念
/*
 注意点:
 1.一定要在自动释放池中 调用 autorelease,才会将对象方法自动释放池中
 2. 在自动释放池中创建了对象,移动要调用 autorelease,才会将对象放到自动释放池中
 3.只要在自动释放池中 调用 autorelease, 就会将对象放入自动释放池中
 4.一个程序中 可以创建N个自动释放池, 并且自动释放池 还可以嵌套
     如果存在多个自动释放池的时候,自动释放池 是以 "栈"的形式存储的
     栈的特点: 先进后出
 */
#pragma mark - 代码
#import <Foundation/Foundation.h>
#pragma mark 类
#import "Person.h"
#pragma mark - main函数
int main(int argc, const char * argv[])
{
    /*
    Person *p = [[Person alloc]init];
    @autoreleasepool {
//        Person *p = [[Person alloc]init];
//        [p run];
        // 2. 在自动释放池中创建了对象,移动要调用 autorelease,才会将对象放到自动释放池中
//        Person *p = [[[Person alloc]init]autorelease];
//        [p run];
        
        // 3.只要在自动释放池中 调用 autorelease, 就会将对象放入自动释放池中
        p = [p autorelease];
        [p run];
    }
    // 1.一定要在自动释放池中 调用 autorelease,才会将对象方法自动释放池中
//    Person *p = [[Person new]autorelease];
     */
    
    // 4.一个程序中 可以创建N个自动释放池, 并且自动释放池 还可以嵌套
    // 如果存在多个自动释放池的时候,自动释放池 是以 "栈"的形式存储的
    // 栈的特点: 先进后出
    
    // 给一个对象方法 一条autorelease消息,永远会将对象 放到栈顶的自动释放池
    @autoreleasepool { // 创建第一个释放池
        @autoreleasepool { // 创建第一个释放池
            @autoreleasepool { // 创建第一个释放池
                Person *p = [[[Person alloc]init]autorelease];
                [p run];
            }// 第三个 释放池 销毁
        }// 第二个 释放池 销毁
    }// 第一个 释放池 销毁
    
    return 0;
}
Person
>>>.h
#import <Foundation/Foundation.h>

@interface Person : NSObject

- (void)run;

@end

>>>.m
#import "Person.h"

@implementation Person

- (void)run
{
    NSLog(@"%s",__func__);
}
- (void)dealloc
{
    NSLog(@"%s",__func__);
    [super dealloc];
}

@end

三、项目2(02-自动释放池大对象问题)
ViewController.m
#import "ViewController.h"
#import "Person.h"
@interface ViewController ()

@end

@implementation ViewController

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    /*
    // 1.不要在自动释放池 使用比较消耗内存的对象,占用内存比较大的对象
    @autoreleasepool {
        Person *p = [[[Person alloc]init]autorelease];
        // 加入p对象 只在100行的地方使用,以后都不用了
        // 一万行代码
    }
     */
    
#warning 比较消耗内存的做法
    // 2.尽量不要在自动释放池使用循环, 特别是循环的次数非常多,并且非常占用内存
    @autoreleasepool {
        for (int i = 0; i<99999; ++i) {
            // 每调用一次 都会创建一个新的对象
            // 每个对象 都会占用一块存储空间
            Person *p =[[[Person alloc]init]autorelease];
        }
    }// 只有执行到这一块,所有的对象才会被释放
    
    
    /*
    for (int i = 0; i<99999; ++i) {
        @autoreleasepool {
            Person *p =[[[Person alloc]init]autorelease];
            // 执行到这一行,自动释放池 就释放了
        }
    }
     */
    NSLog(@"----");
}

@end


栈的分析图

image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 1.Difference between shallow copy and deep copy? 浅复制和深复制的...
    用心在飞阅读 4,617评论 0 9
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,401评论 30 472
  • 内存管理 简述OC中内存管理机制。与retain配对使用的方法是dealloc还是release,为什么?需要与a...
    丶逐渐阅读 6,070评论 1 16
  • 下面是我最近两年学习OC中的一些基础知识,对于学习OC基础知识的人可能有些帮助,拿出来分享一下,还是那句话不喜勿喷...
    小小赵纸农阅读 7,634评论 1 7
  • 问题描述: pop 手势就是为了在大屏下能够获得更好的用户体验设计的。有了 pop 手势,返回的时候不用非要点一下...
    彬至睢阳阅读 5,482评论 3 0

友情链接更多精彩内容