OC内存管理

作者: woooooo | 来源:发表于2016-08-17 09:43 被阅读161次

内存管理的方式

为什么要管理内存



iOS应用程序出现Crash(闪退),90%的原因是因为内存问 题。  
在一个拥有数十个甚至是上百个类的的工程里 ,查找内存问题极其困难,      
学会内存管理,能帮我们减少出错的机率。 
  • 内存问题体现在两个-----内存溢出、野指针异常。

  • 内部溢出

iOS给每个应用程序分配了一定的内存,  
用于程序的运行。一旦超出内存上限,程序就会Crash。
  • 野指针异常
对象的内存已经被系统回收,但是仍然使用指针操作这块内存。  
野指针异常是程序Crash的重要原因之一。  
代码量越大程序越容易出现野指针问题。

内存管理方式

  • 垃圾回收机制(Garbage Collection)。

  • 垃圾回收机制:程序员只需要开辟内存空间,不需要用代码的形式释放,系统来判断哪些空间不再被使用,并回收这些内存空间,以便再次分配。整个回收的过程不需要写任何代码,由系统自动完成垃圾回收。Java开发中一直使用的就是垃圾回收技术。

  • MRC(Manual Reference Counting)。
    -人工引用计数:内存的开辟和释放 都由程序代码进 控制。相对垃圾回收来说,对内存的控制更 加灵活,可以在需要释放的时候及时释放,对程序员的要求较高,程序员要熟悉内存管理的机制。
  • ARC(Auto Reference Counting)。

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

  • ARC是基于MRC的

引用计数

  • C语言中,使用malloc和free,进行堆内存的创建和释放.堆内存只有正在使用和销毁两种状态.
  • 实际开发中,可能会遇到两个以上的指针使用同一块内存。C语言无法记录内存使用者的个数.
OC对象的操作 OC中对应的方法
生成对象 + alloc
持有对象 - retain
释放对象 -release/-autorelease
销毁对象 - dealloc
  • 影响引用计数的方法

    • +alloc: 开辟内存空间,让被开辟的内存空间的引用计数 从0变为1。
    • -retain: 引用计数加1,如果对象之前引用计数为1,retain之后变为2
    • -copy: 把某一对象的内容拷贝一份,拷贝出新的对象, 原有对象的引用计数不变,新的对象的引用计数变1。
    • -release: 引用计数立即减1,如果对象之前的引 计数为 ,release之后变为3,如果之前引用计数为1,release之后变为0,内存被系统回收。
    • -autorelease: 未来的某一时刻引 计数减1 如果对象之前引用计数为4,autorelease之后仍然为4,未来某一刻会变为3。

autoreleasepool的使用

  • 通过autoreleasepool自动释放池,控制autoreleasepool对象的释放。
  • 向一个对象发送autorelease消息,该对象就会被添加到离autorelease最近的自动释放池中,当自动释放池销毁时,为池中的每一个对象发送release消息。

NSAutoreleasePool


iOS5之前,使用NSAutoreleasePool自动释放池类创建对象。  

NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]; // 动释放池创建
[pool release]; // 动释放池销毁

dealloc

  • -dealloc是继承父类的方法,当对象引用计数为0的时候,由对象自动调用 ,销毁该对象的空间。
  • 重写dealloc方法,验证对象的空间是否被回收。
- (void)dealloc {
    
    //这里面写一些本类使用的其他资源的销毁操作,并且需要在父类dealloc方法调用之前
    NSLog(@"调用了销毁");
    //MRC下重写dealloc方法需要调用父类的实现,ARC下重写dealloc方法不需要写[super dealloc],系统会自动调用
    [super dealloc];
}

  • 下面代码会出现什么样的问题
    @autoreleasepool {
        for (NSInteger i = 0; i < 1000000000000; i++) {
            Person *p = [[Person alloc]init];
            [p autorelease];
        }
    }
//因为autorelease 出自动释放池才释放  所以造成内存增加
  • 解决方法

    
    @autoreleasepool {
        for (NSInteger i = 0; i < 1000000000000; i++) {
            @autoreleasepool {
                Person *p = [[Person alloc]init];
                [p autorelease];
            }
        }
    }

    //把一个对象添加到一个集合中会对该对象的引用计数加1,如果移除会将该对象的引用计数减1
    Person *p4 = [[Person alloc] init];
    NSMutableArray *arr = [NSMutableArray array];
    
    [arr addObject:p4];
    [p4 release];//release alloc的+1

内存管理原则

凡是使用了alloc,retain,copy让内存的引用计数 增加了,就需要使release或者autorele让内存的引 计数减少。在一段代码内,增加和减少的次数要相等。
   如果增加的次数多于减少的次数,会造成内存泄露。
   如果增加次数少于减少的次数,会造成过度释放。
   如果增加的次数等于减少的次数,还继续访问,造成野指针问题。

相关文章

  • 内存管理

    一.内存基本介绍 1、OC内存管理的基本概念 2、OC内存管理的范围​管理范围:管理任何继承自NSObject的对...

  • 内存管理

    目录一、内存分区 1、RAM和ROM 2、内存的五大分区二、内存管理 1、OC内存管理是指什么?OC内存管理的本质...

  • Lesson 0-1 Objective-C basic

    6.OC 手动内存管理 OC 内存管理原则: 只要使用 alloc, new, copy, mutableCopy...

  • OC的内存管理

    1、OC的内存管理 OC是通过引用计数进行内存管理的,其核心思想遵循“谁创建谁释放;谁引用谁管理”。 OC的内存管...

  • iOS之从MRC到ARC内存管理详解

    概述 在iOS中开发中,我们或多或少都听说过内存管理。iOS的内存管理一般指的是OC对象的内存管理,因为OC对象分...

  • iOS五大块知识总结之内存管理

    1.1 管理的原因 只有OC对象才需要管理内存,非OC对象(如:char、int、folat)则不需要管理内存的本...

  • iOS 中堆和栈的区别以及冒泡排序

    在引入堆和栈之前,先要知道,iOS中的内存管理范围: OC对象需要进行内存管理,非oc对象不需要进行内存管理,比如...

  • OC的内存管理

    最近系统地复习了OC的内存管理,所以总结下. OC是通过引用计数(reference count)来管理内存.自...

  • OC关于在MRC模式下的内存管理学习

    内存管理 管理范围 任何继承NSObject的对象 只有OC对象才需要进行内存管理的本质原理 1.OC对象在堆中 ...

  • 《Effective Objective-C 2.0》读书笔记(

    第五章 内存管理 第29条:理解引用计数 OC 中有手动内存管理(MRC) 自动内存管理(ARC)手动内存管理需要...

网友评论

  • 红烧大鸡腿:NSAutoreleasePool内程序执行完成时,pool对象立即释放还是延时释放?😊😊😊

本文标题:OC内存管理

本文链接:https://www.haomeiwen.com/subject/eprmsttx.html