详解关于iOS内存管理的规则思考

 更新时间:2016年12月22日 10:21:29   作者:Jsen_Wang  
本篇文章主要介绍了关于iOS内存管理的规则思考,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。

关于iOS内存管理的规则思考

自己生成的生成的对象,自己持有。

非自己生成的对象,自己也能持有。

不在需要自己持有的对象时释放。

非自己持有的对象无法释放。

注:这里的自己是对象使用的环境,理解为编程人员本身也没有错

对象操作和Objective-C方法对应

对象操作 Objectivew-C方法
生成并持有对象 alloc/copy/mutableCopy/new或以此开头的方法
持有对象 retain
释放对象 release
废弃对象 dealloc

自己生成的对象,自己持有

//自己生成并持有对象
id obj1 = [[NSObject alloc] init];

id obj2 = [NSObject new];

id obj3 = [obj2 copy];

copy方法基于NSCopying方法约定,实现类中的copyWithZone:

mutableCopy方法基于NSMutableCopying方法约定,实现类中的mutableCopyWithZone:

非自己生成的对象,自己也能持有

用alloc/new/copy/mutableCopy以外的方法取得的对象,自己不是该对象的持有者。

//取的非自己生成并持有的对象,
//取得对象的存在,但自己不持有对象。

id obj = [NSMutableArray array];

id obj2 = [NSDictionary dictionary];

//自己持有对象
[obj retain];

[obj2 retain];

注:这里有点不好理解,我们先来看一段代码:

//取的非自己生成并持有的对象,
//取得对象的存在,但自己不持有对象。

id unretain_obj = [NSMutableArray array];

NSLog(@"unretain_obj retain count = %lu", (unsigned long)[unretain_obj retainCount]);
//调用 release
[unretain_obj release];

上述代码,我们打印结果是:

2016-12-21 15:32:04.485 acm[65216:852108] unretain_obj retain count = 1

随后调用release方法会导致程序崩溃!

按照引用计数来说,这时unretain_obj是可以被执行一次release方法的。但是为什么我们直接调用会导致程序崩溃。

我们会想最开始提到的四条思想之一:

无法释放非自己持有的对象

这样我们就很好理解了。虽然打印出unretain_obj的retainCount 为 1 但是不能说明是因为它引用了对象。它只是单纯的获取到了对象的存在而已。

那么我们会产生一个问题。那么这个对象是谁在持有??

我们先做一个猜测:

因为[NSMutableArray array]是一个工厂方法,在array肯定是要生成一个NSMutableArray实例对象。这时也必然会有一个指针引用它然后返回这个对象。so。。。

先想到这里,后边我们再去印证

我们再来看一段代码:

//取的非自己生成并持有的对象,
//取得对象的存在,但自己不持有对象。
id unretain_obj = [NSMutableArray array];

NSLog(@"unretain_obj retain count = %lu", (unsigned long)[unretain_obj retainCount]);

//自己持有对象
[unretain_obj retain];

NSLog(@"unretain_obj retain count = %lu", (unsigned long)[unretain_obj retainCount]);

//释放自己持有的对象
[unretain_obj release];

NSLog(@"unretain_obj retain count = %lu", (unsigned long)[unretain_obj retainCount]);

打印结果

2016-12-21 15:40:20.774 acm[65682:861135] unretain_obj retain count = 1
2016-12-21 15:40:20.774 acm[65682:861135] unretain_obj retain count = 2
2016-12-21 15:40:25.254 acm[65682:861135] unretain_obj retain count = 1

并且程序也不会崩溃。

着也印证了我们上边的想法。

因为通过retain方法,非自己生成的对象跟用alloc/new/copy/mutableCopy方法生成并持有的对象一样,成了自己所持有的

不在需要自己持有的对象时释放

通过上边的例子我们知道,自己持有的对象在释放时调用release方法,eg:

//自己生成并持有对象
id release_obj = [[NSObject alloc] init];

//将自己持有的对象释放
[release_obj release];

/* 
 * 释放对象
 * 指向对象的指针依然被保留在变量release_obj 中,你依然可以调用它。
 * 但是对象一经释放绝对不可访问,否则会造成程序崩溃。
 * 出现EXC_BAD_ACCESS Crash问题
 */

我们自己实现一个方法,返回一个方法调用着也可以持有的对象,即alloc的作用

- (id)allocObject {
   //自己生成并持有对象
  id obj = [[NSObject alloc] init];
  //原封不动的返回一个由alloc方法生成的对象
  return obj;

注:方法名符合 生成并持有对象  alloc/copy/mutableCopy/new或以此开头的方法 规则

我们自己实现一个方法,返回一个谁也不持有的对象,只是取得对象的存在

- (id)object {
  //自己生成并持有对象
  id obj = [[NSObject alloc] init];

  //调用autorelease方法 取得对象的存在,但自己不持有对象。
  [obj autorelease];

  return obj;

autorelease方法可以取得对象的存在,但自己不持有对象。使对象在超出指定的生存范围时能够自动的并正确的释放(调用release方法)

autorelease和release方法的区别

autorelease:

release:

autorelease的详细解说我们后边介绍。

我们也可以通过调用retain方法来使 autorelease方法的来的对象自己持有eg:

//获取对象的存在,自己不持有
 id unretain_obj = [NSMutableArray array];

 //持有对象
[unretain_obj retain];

无法释放非自己持有的对象

自己已经释放了还继续释放

  //自己生成并持有对象
  id release_obj = [[NSObject alloc] init];

  //将自己持有的对象释放
  [release_obj release];

  //释放已经释放的对象
  [release_obj release];

  /*
   * 释放对象
   * 指向对象的指针依然被保留在变量release_obj 中,你依然可以调用它。
   * 但是对象一经释放绝对不可访问,否则会造成程序崩溃。
   * 出现EXC_BAD_ACCESS Crash问题
   */

只获取了对象的存在,试图释放对象

  //取的非自己生成并持有的对象,
  //取得对象的存在,但自己不持有对象。
  id unretain_obj = [NSMutableArray array];
  //释放自己不持有的对象
  [unretain_obj release];

程序崩溃,报EXC_BAD_ACCESS Crash问题

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • iOS实现拼图小游戏

    iOS实现拼图小游戏

    这篇文章主要为大家详细介绍了iOS实现拼图小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • iOS状态栏、导航栏的一些笔记分享

    iOS状态栏、导航栏的一些笔记分享

    这篇文章主要给大家分享了关于iOS中状态栏、导航栏的一些笔记,文中通过示例代码介绍的非常详细,对各位iOS开发者们具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-04-04
  • iOS10适配以及Xcode8使用需要注意的那些坑

    iOS10适配以及Xcode8使用需要注意的那些坑

    这篇文章主要为大家详细介绍了iOS10的适配以及Xcode8使用需要注意的那些坑,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-09-09
  • IOS之构造方法与自定义构造方法的区别与实现

    IOS之构造方法与自定义构造方法的区别与实现

    本篇文章主要介绍了构造方法以及自定义构造方法的实现,需要的朋友可以参考下
    2015-07-07
  • 通过一行代码搞定UITextField的输入格式限制

    通过一行代码搞定UITextField的输入格式限制

    这篇文章主要给大家介绍了如何通过一行代码搞定UITextField的输入格式限制的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-08-08
  • iOS页面跳转及数据传递(三种)

    iOS页面跳转及数据传递(三种)

    本文主要介绍了iOS页面跳转的三种方法及数据传递的方法。具有很好的参考价值。下面跟着小编一起来看下吧
    2017-03-03
  • 阿里数据iOS端启动速度优化心得

    阿里数据iOS端启动速度优化心得

    本篇文章给大家详细分析了阿里数据iOS端启动速度优化的知识点以及心得,对此有兴趣的朋友参考学习下吧。
    2018-02-02
  • iOS11&iPhoneX适配&Xcode9打包注意事项

    iOS11&iPhoneX适配&Xcode9打包注意事项

    这篇文章主要介绍了iOS11&iPhoneX适配&Xcode9打包注意事项,需要的朋友可以参考下
    2017-10-10
  • iOS使用UIScorllView实现两指缩放功能

    iOS使用UIScorllView实现两指缩放功能

    两指缩放功能不仅可以用UIPinchGestureRecognizer手势来实现,还能用UIScorllView来实现,UIScrollView可以轻松的实现最大与最小缩放值,以及滚动的效果,效果非常棒,具体实例代码大家参考下本文吧
    2017-03-03
  • 详解iOS tableViewCell自适应高度 第三发类库

    详解iOS tableViewCell自适应高度 第三发类库

    在github中有许多大牛封装好的第三发类库,其中有个自适应cell高度的类库。接下来通过本文给大家介绍iOS tableViewCell自适应高度 第三发类库,需要的朋友参考下
    2016-04-04

最新评论