您的位置:首页 > 移动开发 > IOS开发

iOS内存管理篇(一)---alloc/reatain/release/dealloc方法实现

2017-06-28 18:40 591 查看

前言:自从苹果开发出 ARC这个后,基本上使用 MRC开发的就很少了,但是我们还是有必要了解一下原理以及过去的使用。

1. MRC是什么

MRC(MannulReference Counting)简而言之就是手动计数,手动管理 对象的释放以及引用计数

2. ARC是什么

ARC(Automatic Reference Counting) 自动管理计数,不需要写多余的代码。

我们简单的来了解一下MRC的工作

现在我们来新建一个工程 并且在如图的地方输入 “-fno-objc-arc”



然后我们在 appdelegate.m文件里面输入如下的代码

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {

NSObject *object=[[NSObject alloc]init];
///这个时候 object 没有被别的对象持有 仅仅调用了 [object retainCount] 方法 所以 object的引用计数为1
NSLog(@"objectCount:%ld",[object retainCount]);

NSObject *another=[object retain];
///这个时候 object 被 another 持有 所以引用计数+1 此时引用计数为 2
NSLog(@"objectCount:%ld",[object retainCount]);

[another release];
///执行完[another release]这句代码后 another 的被释放 所以 object 的引用计数变成1了
NSLog(@"objectCount:%ld",[object retainCount]);

///执行完这句代码后, object 引用计数就为0了
[object release];
return YES;
}


输出结果如下:



我们通过这个例子可以看出来,对于 MRC来说,需要手动维护对象的计数,管理对象的存在和销毁的问题.如果稍有不注意的地方,就会存在内存泄露和循环引用的问题,那么苹果为了避免这一套繁琐的开发任务,方便开发者去更好的进行开发任务,于是开发出了一套 ARC,这是我们今天重点学习的内容

在此之前,我们需要很好的了解什么引用计数:

一个对象被持有的数量,打个比方来说,有一根绳子,没有人握住的时候,引用计数是0,当1个人握住的时候 引用计数+1 ,往后每叠加一个人,引用计数变依次+1,当有一个人松手的时候引用计数-1,直到没有绳子没有人握住的时候,绳子掉下来 销毁了,这就是引用计数的概念.

所以引用计数的管理方法是

每个对象都有一个与之关联的整数,这个整数被称为引用计数,在Objective-C中,通过不同的方法可以对引用计数进行操作,具体的处理如下表:

对象操作Objective-C方法对应的操作结果
生成并持有对象alloc, new, copy,mutableCopy等方法生成对象并设置引用计数 =1
持有对象reatain方法使引用计数 +1
释放对象release方法使引用计数 -1
废弃对象dealloc方法—系统自动调用引用计数 =0 时调用
来讲一下 alloc/reatain/release/dealloc方法的实现

alloc

我们看一下 alloc 如何实现的

+(id)alloc
{
return [self allocWithZone:NSDefaultMallocZone()];
}
+(instancetype)allocWithZone:(struct _NSZone *)zone
{
return NSAllocateObject(self, 0, zone);
}


通过allocWithZone类方法调用 NSAllocateObject方法来开辟了一块内存空间,我们来看一下NSAllocateObject方法是如何实现的

struct obj_layout{
NSUInteger retained;
};

inline id
NSAllocateObject(Class  _Nonnull aClass, NSUInteger extraBytes, NSZone * _Nullable zone)
{
int size = 计算容纳对象所需要的内存大小
id new = NSZoneMalloc(zone, size);
memset(new,0,size);
new = (id)&((struct obj_layout *)new)[1];

}


NSAllocateObject通过调用NSZoneMalloc函数分配存放对象所需要的内存控件,之后将该内存空间置0,最后返回作为对象而使用的指针

我们可以执行一下如下代码

id obj = [[NSObject alloc]init];
NSLog(@"retainCount = %lu",(unsigned long)[obj retainCount]);


执行结果为:

2017-06-28 18:15:04.009 MRCTest[12244:1459486] retainCount = 1


reatian 和 release 的方法刚刚在 MRC已经看过 我们看一下 dealloc 方法

-(void)dealloc
{
NSDeallocateObject(self);
}

inline void
NSDeallocateObject(id  _Nonnull object)
{
struct obj_layout *o = ((struct obj_layout*) anObject)[-1];
free(o);
}


每个对象都具备 dealloc 方法,当一个对象的引用计数为0的时候,也就意味着没有任何地方需要该对象,系统会自动回收对该对象所占用的内存,在系统销毁对象的时候,会自动调用该对象的 dealloc 方法来执行一些回收的操作,如果此时该对象还对其他对象有引用的话,那么就需要重写 dealloc 方法来释放该对象对其他对象的引用 以确保该对象能正常释放销毁

如何重写 dealloc 方法

- (void)dealloc {

// 处理该对象的其他引用(通过release方法)

/** 回调父类的dealloc方法 */
[super dealloc];
}


对于alloc/reatain/release/dealloc可以总结如下:

在 Objective-C的对象中存有引用计数这一整数值

调用 alloc或者是 retain 方法后,引用计数值加1

调用 release 后,引用计数值减1

引用计数为0时,调用 dealloc 方法销毁此对象

苹果对这四个方法的使用

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

__CFDoExternRefOperation
CFBasicHashRemoveValue      // 当此函数返回0时, -release调用dealloc方法


可以从__CFDoExternRefOperation函数以及由此函数调用的哥哥函数名来看,苹果的实现大概就是采用的引用计数表来管理引用计数的

如图



好了今天这几个方法就讲到这里,明天我们将来学习一下AutoreleasePool
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  ios 内存管理 对象
相关文章推荐