1.什么是内存管理

  • 移动设备的内存极其有限,每个app所能占用的内存是有限制的
  • 当app所占用的内存较多时,系统会发出内存警告,这时得回收一些不需要再使用的内存空间。比如回收一些不需要使用的对象、变量等
  • 管理范围:任何继承了NSObject的对象需要去管理内存,但是对于对其他基本数据类型(int、char、float、double、struct、enum等)结构,枚举等不用去关心内存
- (void)test
{
int a = 2;
int b = 1; Person *person = [[Person alloc] init]; // 方法未退出之前 这三行代码在内存中如图
}

一旦test方法执行完毕,意味着局部变量的作用域也失效,那么栈空间的局部变量系统会自动检测回收。但是堆空间中动态产生的对象是还没有被回收。如图

可以看出即使已经没有指针指向动态产生的对象了,但还是没有被回收,因此需要手动管理释放。释放的方法是为对象发送一条消息。因此需要调用对象的某个方法来释放对象。

那么系统是怎么知道此时的对象需不需要回收呢?这就涉及到了对象结构中的"引用计数"

2.对象结构

  • 每个OC对象内部都有自己的引用计数器,它是一个整数,表示"对象被引用的次数",即有多少人正在使用这个OC对象
  • 每个OC对象内部会自动设置4个字节的存储空间来存储引用计数器

3.引用计数器的作用

  • 当使用alloc、new或者copy创建一个新对象时,新对象的引用计数器默认就是1
  • 当一个对象的引用计数器值为0时,对象占用的内存就会被系统回收。换句话说,如果对象的计数器不为0,那么在整个程序运行过程,它占用的内存就不可能被回收,除非整个程序已经退出

4.操作对象引用计数器的方法

  • 给对象发送一条retain消息,可以使引用计数器值+1(retain方法返回对象本身)retain方法返回的是id类型,那么哪个对象调用返回的就是自己
  • 给对象发送一条release消息,可以使引用计数器值-1
  • 可以给对象发送retainCount消息获得当前的引用计数器值

 5.对象的销毁

  • 当一个对象的引用计数器值为0时,那么它将被销毁,其占用的内存被系统回收
  • 当一个对象被销毁时,系统会自动向对象发送一条dealloc消息
  • 一般会重写dealloc方法,在这里释放相关资源,dealloc就像对象的遗言
  • 一旦重写了dealloc方法,就必须调用[super dealloc],并且放在最后面调用
  • 不能直接调用dealloc方法
  • 一旦对象被回收了,它占用的内存就不再可用,坚持使用会导致程序崩溃(野指针错误)
    // alloc方法是给堆中分配内存 init方法和内存无关 此时retainCurrent为1
Person *p = [[Person alloc] init]; // 返回的就是对象本身 retainCurrent为2
[p retain]; // retainCurrent为1
[p release]; // retainCurrent为0 说明Person类对象被回收,那么对应的在内存中的地址已经不可用了 此时的Person对象称为“僵尸对象”但是此时p指针还是在指向Person类对象所对应的那块不可用的地址此时的p指针称为“野指针”
[p release];

 6.开发中要注意的内存管理

默认情况下,Xcode是不会管理僵尸对象的,即使使用了一块被释放的内存也不会报错。为了方便调试,应该开启僵尸对象监控。如图设置:

注意三个概念:

僵尸对象:已经被回收的对象,或者说对象所对应的内存地址已经不可用的对象称为僵尸对象。僵尸对象不可用

野指针:指向一块不可用内存地址或者指向僵尸对象的指针称为野指针。给野指针发送消息会报 EXC_BAD_ACCESS错误

空指针:没有指向任何指针变量称为空指针,也意味着指针变量所存储的值为0,nil,NULL 这样可以避免野指针错误的发生

/********************************** Person.h **************************************/
#import <Foundation/Foundation.h> @interface Person : NSObject @property int age; @end /********************************** Person.m **************************************/
#import "Person.h" @implementation Person // 重写父类NSOjbct的遗言方法 对象在被释放之前一定会调用dealloc方法 - (void)dealloc
{
NSLog(@"对象在释放之前会执行遗言方法被执行"); [super dealloc]; // 一定要调用 而且必须放在最后面
}
@end /********************************** main.m **************************************/
#import <Foundation/Foundation.h>
#import "Person.h" /*
main方法是一个死循环方法以保证程序能持续运行,除非用户关闭程序或者是手机没电,程序才能终止
那么在main方法里面的Person对象不就一直存在么,因此必须在main方法里面将对象回收
*/
int main(int argc, const char * argv[])
{
// alloc方法是给堆中分配内存 init方法和内存无关 此时retainCurrent为1
Person *p = [[Person alloc] init]; // 返回的就是对象本身 retainCurrent为2
[p retain]; // retainCurrent为1
[p release]; /*
retainCurrent为0
说明Person类对象被回收,那么对应的在内存中的地址已经不可用了
此时的Person对象称为“僵尸对象”
此时p指针还是在指向Person类对象所对应的那块不可用的地址,此时p指针称为“野指针”
*/
[p release]; /*
对象已经被回收,千万别以为再给对象发送一个retain消息对象就可以“起死回生”应该节哀顺变
执行retain方法会报错,此时的p指针已称为野指针执行代码回报:野指针错误
*/ // [p retain]; /*
此时对象已经被回收称为“僵尸对象了”不可以再访问属性
在执行p.age = 10;报错:
-[Person setAge:]: message sent to deallocated instance 消息发送给了已经被释放的对象
再次证明“僵尸对象不可以用”
*/ /*
一旦指针成为野指针再继续向p指针所指的对象发送消息就会报错:Exc_BAd_ACCESS
说明访问了一块坏内存(已经被回收、不可用的内存) “野指针错误”
那么此时在对象回收之后将指针变量清空
那么栈中的指针变量就不会再指向堆中类对象的内存地址了
*/
p = nil; /*
指针变量内部所存储的值已被清空,那么指针已经无指向
再给指针发送任何消息指针会无任何响应,而且也不报错因为OC中没有空指针错误
*/
[p release];
[p release];
[p release];
[p release]; return ;
}

iOS 非ARC基本内存管理系列 1-引用计数器的更多相关文章

  1. iOS 非ARC基本内存管理系列 -手把手教你ARC——iOS/Mac开发ARC入门和使用(转)

    手把手教你ARC——iOS/Mac开发ARC入门和使用 Revolution of Objective-c 本文部分实例取自iOS 5 Toturail一书中关于ARC的教程和公开内容,仅用于技术交流 ...

  2. iOS 非ARC基本内存管理系列 2-多对象内存管理(3) 利用@property来自动管理内存

    iOS 基本内存管理-多对象内存管理(2)中可以看到涉及到对象的引用都要手动管理内存:每个对象都需要写如下代码 // 1.对要传入的"新车"对象car和目前Person类对象所拥有 ...

  3. iOS 非ARC基本内存管理系列总结6 -设计微博模型

    设计简单的微博模型:用User类和Status类来模拟实现 在非ARC机制下有两种方式,两者没有太大的区别之所以写了两种只是为了方便学习和对比两种写法! 第一种:没有使用atuorelease和自动释 ...

  4. iOS 非ARC基本内存管理系列 4-autorelease方法和@autoreleasepool

    1.autorelease 基本用法 对象执行autorelease方法时会将对象添加到自动释放池中 当自动释放池销毁时自动释放池中所有对象作release操作 对象执行autorelease方法后自 ...

  5. iOS 非ARC基本内存管理系列 2-多对象内存管理(2)

    /* 多对象内存管理: 以人拥有车为例涉及到@property底层set方法管理内存的实现 注意:人在换车的时候要进行当前传入的车和人所拥有的车进行判断 */ /******************* ...

  6. iOS 非ARC基本内存管理系列 2-多对象内存管理(1)

    单个对象的内存管理非常简单无非就是alloc对应release,retain对应release.但是如果涉及到很多对象,而且对象与对象有联系的时候该怎么去管理对象的内存呢. 比如同样一本书有好3个人购 ...

  7. iOS 非ARC基本内存管理系列 5-autorelease方法使用总结

    autorelase:可以将对象交给自动释放池中,释放池销毁的时候对里面的对象做一次release操作代码如下 @autoreleasepool { Person *person = [[[Perso ...

  8. iOS 非ARC基本内存管理系列 3-循环retain和@class

    1.@class 使用场景:对于循环依赖关系来说,比方A类引用B类,同时B类也引用A类: 可以看出Person和Card互相引用,此时如果使用#import编译报错!因此当使用@class在两个类中相 ...

  9. IOS 非ARC开发内存管理的几条规则

    关于ios内存管理.在开发过程中,内存管理很重要,我简单说明一下. 1.正确用法 UIView *v = [[UIView alloc] init]; //分配后引用计数为1 [self.view a ...

随机推荐

  1. 使用安卓读取sqlite数据库方法记录

    最近要实现android读取sqlite数据库文件,在这里先做一个英汉字典的例子.主要是输入英语到数据库中查询相应的汉语意思,将其答案输出.数据库采用sqlite3. 如图: 实现过程完全是按照参考文 ...

  2. Android 实现书籍翻页效果----完结篇

    By 何明桂(http://blog.csdn.net/hmg25) 转载请注明出处 之前由于种种琐事,暂停了这个翻页效果的实现,终于在这周末完成了大部分功能,但是这里只是给出了一个基本的雏形,没有添 ...

  3. oc-08-内存分析

    说有对象公用类的一个方法

  4. C读取配置文件

    #ifndef __CFG_OP_H__ #define __CFG_OP_H__ #ifdef __cplusplus extern "C" { #endif //获取配置项 i ...

  5. 操作cookie

    $.extend($, { setCookie: function(c_name, value, expiredays) { var exdate = new Date(); exdate.setDa ...

  6. Gradle Goodness: Set Java Compiler Encoding--转载

    原文地址:http://java.dzone.com/articles/gradle-goodness-set-java If we want to set an explicit encoding ...

  7. 小白日记28:kali渗透测试之Web渗透-扫描工具-Nikto

    扫描工具-Nikto #WEB渗透 靶机:metasploitable 靶场:DVWA[默认账号/密码:admin/password] #新手先将DVWA的安全性,调到最低,可容易发现漏洞 侦察[减少 ...

  8. iCloud之旅

    1.创建BIDTinyPixDocument类 #import <UIKit/UIKit.h> //创建文档类 @interface TinyPixDocument : UIDocumen ...

  9. U盘安装Debian 7

    网上看到好多使用U盘安装的教程,齐说不一.实践是检验真理的标准,Try it ! 下载系统镜像:http://cdimage.debian.org/cdimage/release/current/am ...

  10. xcode笔记

    1.Alt键的使用   2.设置捕捉所有意外断点:停在代码出错处     2015年07月27日09:52:12 3.搜索 command + F:在当前的文件中搜索 command + Shift ...