Objective-C内存管理之-引用计数
本文会继续深入学习OC内存管理,内容主要参考iOS高级编程,Objective-C基础教程,疯狂iOS讲义,是我学习内存管理的笔记
内存管理
1 内存管理的基本概念
1.1 Objective-C中的内存管理
- 手动内存管理和自动释放池---
MRC
>(Mannul Reference Counting) - 自动内存管理---
ARC
>(Automatic Reference Count) - 自动垃圾回收---
GC
>(Garbage Collection)
由于iOS系统不支持垃圾回收,所以我们在iOS开发中只能使用MRC和ARC来进行内存管理,本文不再介绍Objective-C中的垃圾回收机制,但是此处注意Objective-C中是存在垃圾回收机制的
1.2 内存管理中存在的问题
内存泄露
:不再需要的对象没有释放引起的问题
:程序的内存占有量不断增加,最终会被耗尽导致程序崩溃野指针
:没有进行初始化得指针引起的问题
:浪费内存资源,如果调用程序会出现未知的结果,甚至导致程序崩溃悬空指针
:一个指针指向一个被销毁的对象引起的问题
:调用悬空指针指向的属性或者方法时,程序会出现未知的结果,甚至导致程序崩溃僵尸对象
:过度释放的对象引起的问题
:
2.手动内存管理和自动释放池---MRC
>(Mannul Reference Counting)
2.1 什么是引用计数(Reference Counting)
引用计数
:Objective-C中引入了引用计数这一机制来跟踪并处理对象的生命周期,
管理方式
:每个对象都有一个与之关联的整数,这个整数被称为引用计数,在Objective-C中,通过不同的方法可以对引用计数进行操作,具体的处理如下表:
对象操作 | Objective-C方法 | 对应的操作结果 |
---|---|---|
生成并持有对象 | alloc , new , copy ,mutableCopy 等方法 |
生成对象并设置引用计数 =1 |
持有对象 | reatain 方法 |
使引用计数 +1 |
释放对象 | release 方法 |
使引用计数 -1 |
废弃对象 | dealloc 方法---系统自动调用 |
引用计数 =0 时调用 |
关于delloc
方法:dealloc方法继承自NSObject
,因此所有的对象都具有此方法,当一个对象的引用计数为0时,也就意味着没有任何程序需要此对象,系统会回收该对象所占用的内存,在系统销毁对象之前,会自动调用该对象的dealloc方法来执行一些回收操作,如果该对象还持有其他对象的引用,我们必须重写dealloc方法来释放该对象引用的其他对象(通常就是使用该对象的release方法)
引用计数机制回收对象的说明
:如果一个对象的引用计数为0,则表明程序已经不再需要它,这时系统会自动回收该对象所占内存,相反,如果一个对象的引用计数不为0,系统就不应该回收,也不会回收它所占的内存
关于retainCount方法
:Objective-C提供了retainCount方法来返回一个对象当前的引用计数
如何重写dealloc方法
:
- (void)dealloc {
// 处理该对象的其他引用(通过release方法)
/** 回调父类的dealloc方法 */
[super dealloc];
}
2.2 苹果如何管理引用计数
- 2.2.1 因为NSObject类的源代码没有公开,我们利用Xcode的调试器(lldb)和iOS大概追溯出其实现过程
alloc
+alloc
+allocWithZone:
class_createInstance //此方法可以通过objc4中的runtime/objc-runtime-new.mm确认
calloc // 分配内存块
retainCount
-retainCount
__CFDoExternRefOperation // 此函数根据retain,retainCount,release操作进行分发,调用__CFBasicHashXXX方法
CFBasicHashGetCountOfKey
retain
-retain
__CFDoExternRefOperation
CFBasicHashAddValue
release
-release
__CFDoExternRefOperation
CFBasicHashRemoveValue // 当此函数返回0时, -release调用dealloc方法
- 2.2.2 由__CFDoExternRefOperation函数以及此函数的调用关系,我们大概推算苹果大概是使用散列表(引用计数表)来管理引用计数
- 通过引用计数表来管理引用计数的好处:
- 对象用内存块的分配无须考虑内存块头部
- 引用计数表各记录中存有内存块的地址,可从各个记录追溯到各对象的内存块(在进行内存泄露的检查时,此条特性具有举足轻重的作用,即使出现故障导致对象占用的内存块损坏,但是只要引用计数表没有被破坏,我们就可以确定各内存块的位置,这就是设置全局断点可以查出哪里出现内存泄露的原因)
- 通过引用计数表来管理引用计数的好处:
2.3 内存管理的思考方式
自己生成的对象,自己持有
1.1 使用
alloc
new
copy
mutableCopy
创建的对象只能自己持有id obj1 = [[NSObject alloc] init];
id obj2 = [NSObject new];
id obj3 = [NSObject copy];
id obj4 = [NSObject mutableCopy];
1.2 使用以上名称的开头的方法也意味着自己生成并持有对象
alloc
NewObjectnew
NewObjectcopy
NewObjectmutableCopy
NewObject非自己生成的对象,自己也能持有
2.1 非
alloc
new
copy
mutableCopy
生成的对象,变量obj本身不持有该对象id obj1 = [NSMutableArray array];
id obj2 = [NSDictionary dictionary];2.2 通过retain方法,非通过
alloc
new
copy
mutableCopy
生成的对象,可以成为自己持有的对象id obj = [NSMutableArray array]; [obj retain];
不再需要自己持有的对象时释放
3.1 释放通过
alloc
new
copy
mutableCopy
生成的对象,一旦不在需要,务必要使用release方法释放id obj = [[NSObject alloc] init]; [obj release];
3.2 用retain方法持有的非自己生成的对象,一旦不再需要,也一定要使用release释放
id obj = [NSMutableArray array];
[obj retain]; // 通过retain方法持有对象
[obj release]; // 在不需要时也要通过release方法释放对象
3.3 用某个方法生成对象,并将其作为方法的返回值,这时我们该如何处理
3.3.1 通过
alloc
new
copy
mutableCopy
或其他符合命名规则的方法生成的对象,只需要原封不动的返回就能让调用方也持有该对象- (id)allocObject {
id obj = [[NSObject alloc] init];
return obj;
} - (id)allocObjectWithObject:(id)obj {
id object = [obj allocObject];
return object;
}
3.3.2 如果持有非自己生成的对象,例如[NSMutableArray array]生成的对象,我们要使用autorelease方法释放
注:
命名规则
:用来取得谁都不持有的对象的方法名不能以alloc
new
copy
mutableCopy
开头- (id)object {
id obj = [NSMutableArray array];
[obj autorelease];
return obj;
}
3.3.3
autorelease
方法:提供了这样的功能,使对象在超出指定的生存范围时自动并正确释放(调用release方法)
非自己持有的对象无法释放---注意以下两点,如果发生这样的情况会导致程序崩溃
4.1 通过
alloc
new
copy
mutableCopy
方法或者通过retain方法持有的对象,一旦不再需要时,必须进行释放,除此之外其他方法获得的对象绝对不能释放,一旦释放会造成程序崩溃4.2 自己持有的对象释放后再次释放,造成僵死对象,引起程序崩溃或在访问废弃的对象时崩溃
id obj = [[NSObject alloc] init];
[obj release];
[obj release]; // 再次释放
Objective-C内存管理之-引用计数的更多相关文章
- iOS的内存管理和引用计数规则、Block的用法以及三种形式(stack、malloc、global)
学习内容 iOS的内存管理和引用计数规则 内存管理的思考方式 自己生成的对象自己持有 非自己生成的对象自己也能持有 自己持有的对象不需要时释放 非自己持有的对象不能释放 ARC有效时,id类型和对象类 ...
- Objective-C内存管理之引用计数
初学者在学习Objective-c的时候,很容易在内存管理这一部分陷入混乱状态,很大一部分原因是没有弄清楚引用计数的原理,搞不明白对象的引用数量,这样就当然无法彻底释放对象的内存了,苹果官方文档在内存 ...
- Swift基础语法-内存管理, 自动引用计数
1. 工作机制 Swift和OC一样,采用自动引用计数来管理内存 当有一个强引用指向某一个对象时,该对象的引用计数会自动+1 当该强引用消失时,引用计数会自动-1 当引用计数为0时,该对象会被销毁 2 ...
- Python内存管理及引用计数
作为一门动态语言,python很重要的一个概念就是动态类型,即对象的类型和内存占用都是运行时确定的.(Why?)运行时,解释器会根据语法和右操作数来决定新对象的类型.动态类型的实现,是通过引用和对象的 ...
- Objective C内存管理之理解autorelease------面试题
Objective C内存管理之理解autorelease Autorelease实际上只是把对release的调用延迟了,对于每一个Autorelease,系统只是把该Object放入了当前的A ...
- Object-C内存管理-对象引用计数的特例
看到OC中内存管理这块,其中的引用计数部分,部分10.5上的EBOOK示例已经在10.9上不能运行正确了,比如下面的代码: NSString * str1 = @"string 1" ...
- Objective C 内存管理[转]
1 配对原则 alloc – release new – release retain - release copy – release 2 new和alloc-init的区别 (1)区别只在于a ...
- objective C 内存管理及属性方法具体解释
oc为每一个对象提供一个内部计数器.这个计数器跟踪对象的引用计数,当对象被创建或拷贝时.引用计数为1.每次保持对象时,调用retain接口.引用计数加1.假设不需要这个对象时调用release,引用计 ...
- OC语法6——内存管理之引用计数器(retain,release)
OC内存管理: 一.引用计数器: Java有垃圾回收机制(Garbage Collection,GC).也就是说当我们创建对象后,不需要考虑回收内存的事,Java的垃圾回收机制会自动销毁该对象,回收它 ...
随机推荐
- Android使用TextureView播放视频
1.引言 如果你想显示一段在线视频或者任意的数据流比如视频或者OpenGL 场景,你可以用android中的TextureView做到. 1).TextureView的兄弟SurfaceView 应用 ...
- ASP.NET MVC中给所有的cshtml页面引用命名空间
在web.config文件中加入:这样所有需要以下命名空间的页面就不需要再它页面中单独引用这些命名空间了 <system.web.webPages.razor> <host fact ...
- 45 个非常有用的 Oracle 查询语句
这里我们介绍的是 40+ 个非常有用的 Oracle 查询语句,主要涵盖了日期操作,获取服务器信息,获取执行状态,计算数据库大小等等方面的查询.这些是所有 Oracle 开发者都必备的技能,所以快 ...
- MySQL密码忘记,怎么办?
如果哪天你忘记了线上MySQL数据库的root密码,怎么办? 大家往往会想到skip-grant-tables参数,具体步骤如下: 1. 关闭MySQL数据库,因为root密码忘记了,mysqladm ...
- CSS字体记录
aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAaYAAACnCAIAAADVOG9FAAAgAElEQVR4nOy9eXwcxZk/vL/9ve8eb7 ...
- 归一化变换 Normalizing transformations
归一化变换包含两个部分,图像坐标的平移和尺度的缩放.进行归一化的变换不但能够提高处理结果的精确度,而且通过选择一个标准的坐标系预先的消除了图像尺度和坐标原点的选择对算法最终结果的影响. 归一化变换的步 ...
- C#使用资源文件的方法
其实,对于资源文件的使用,说白点就是通过强制类型转换,将资源文件里的数据强行的转换成你需要的(换种方式说,就是你原来存进去什么,就用什么类型拿出来). 主要通过System.Resources.Res ...
- .NET 对象生命周期
GC 垃圾回收 .NET Framework 的垃圾回收器管理应用程序的内存分配和释放.每次您使用 new 运算符创建对象时,运行库都从托管堆为该对象分配内存.只要托管堆中有地址空间可用,运 ...
- IDE有毒
程序员按项目性质大致有三种:写Demo的.写Proto的.写成品的:按项目开发周期大致有:写开头的.写中间的.写结尾的. Demo是样品,主要是表面上初步实现,临时忽悠客户用的,不一定要求继续演化: ...
- error C2664: 'BOOL (PCERT_SELECT_STRUCT_A)' : cannot convert parameter 1 from 'CERT_SELECT_STRUCT *' to 'PCERT_SELECT_STRUCT_A'
1. 编译c++程序出现标题所示错误 2. 解决方法:更改编码设置