现在大家的项目应该基本都是ARC了,如果还是MRC的话,赶紧转换到ARC吧!最近被临时拉过去开发iPad,由于项目原因,还是使用的MRC。今天在调部分界面的时候,发现一段代码,我怎么看都怎么觉得怪怪的,因为是MRC嘛!所以我心里还是一直提醒着自己。仔细一看还真是不对,这段代码给周围同事看的时候,也不是每个人都能一眼看出问题,因为大家已经习惯了ARC或者没有在MRC下进行开发过。

下面我贴出类似的代码:

- (void)pushVc
{
pushVC = [[UIViewController alloc] init]; [self.navigationController pushViewController:pushVC animated:YES];
}

以上这段代码很简单,就是有个UIViewController类型的成员变量pushVC,然后创建一个VC赋值给他,最后push到这个页面。可能很多人一看,这代码就是平常自己写的啊,都没有出现过问题啊。如果这段代码是在ARC下,是没有任何问题的。但是,如果我们的代码是在MRC下,会出现什么问题呢?如果经历过MRC开发的人,肯定也会觉得这边怪怪的。至少没有发现调用release。由于pushVC是成员变量,所以一定程度上也迷惑了下同事。上面的代码其实已经内存泄露了。[[UIViewController alloc] init] 这个方法创建出来的对象将不会被销毁,一直留在内存中。为什么?这个对象创建出来的时候引用是1,然后经过push引用计数已经变成2了。当这个vc在后面被pop出来的时候,引用计数会减1,这时这个VC的引用计数还是1。在内存中将销毁不掉。如果这个方法被多次调用的话,将会出现大量的这个对象在内存中。

下面再说一个知识点,很多人知道,但是并不一定完全了解我们的@property到底做了什么。


- (void)pushVc
{
self.pushVC = [[UIViewController alloc] init]; [self.navigationController pushViewController:pushVC animated:YES];
}

在看这段代码,我给成员变量赋值的方式换成了self.pushVC,这个和直接赋值有什么区别呢?如果你调用self.pushVC进行赋值,那么这个时候会调用系统为我们默认生成的setter方法。这个setter会帮我们做内存的引用计数操作。看下系统生成的方法示例:

- (void)setPushVC:(UIViewController *)setPushVC
{
[setPushVC retain];
[pushVC release];
pushVC = setPushVC;
}

首先,系统会将传进来的对象引用计数加1,之后将赋值的对象引用计数减1,最后再给对象赋值。记得自己重写setter方法的时候,一定要先将传进来的对象做下retain操作,之后在release本身的对象。如果你代码这样写的话:

- (void)setPushVC:(UIViewController *)setPushVC
{
[pushVC release];
[setPushVC retain];
pushVC = setPushVC;
}

正常情况下是没有问题的,但是如果是自己给自己赋值的话self.pushVC = pushVC,那就有问题了。当然你可以做下if判断,两个对象是否一样,那样也行。

接下来看下这个代码的正确写法:

    UIViewController *VC = [[UIViewController alloc] init];
pushVC = [VC retain];
[VC release];
[self.navigationController pushViewController:VC animated:YES];
//或者
UIViewController *VC = [[UIViewController alloc] init];
self.pushVC = VC
[VC release];
[self.navigationController pushViewController:VC animated:YES];

建议大家在MRC下使用成员变量的时候最好使用self.setter方法。有同事又提出了另一种写法:

    pushVC = [[[UIViewController alloc] init] autorelease];

    [self.navigationController pushViewController:pushVC animated:YES];

autorelease,但是这样写有个问题,一旦你使用这个关键字,那你就不在有这个创建对象的内存管理权,系统会在之后的某个时间,对其进行release操作。这样也违背了用成员变量保存这个VC的意图。

总结

很多同事一眼没有看出来,是因为我们已经习惯了ARC,认为=就是给对象进行了retain。在ARC下默认变量前面都有一个隐藏的__strong。在ARC下只要变量指向对象,那么系统会我们自动的对那个对象进行retain操作,当我们将对象置为nil的时候,系统会默认给我们做release操作。

引用计数内存管理的思考方式:

  • 自己生成的对象,自己所持有
  • 非自己生成的对象,自己也恩能持有
  • 自己持有的对象不再需要时释放
  • 非自己持有的对象无法释放

当我们使用ARC的时候,也是遵循了上面的思考方式。不要因为我们没有看到retain或者release而认为管理方式变了或者不需要内存管理了。ARC看起来很简单,因为苹果把那些引用计数的操作代码都交给了编译器,所以给了我们这种错觉。了解MRC,可以加深自己对ARC的理解。不至于让自己被ARC给蒙蔽了。

使用ARC可以让我们的代码更加精简,健壮,特别是weak这个关键字,更是解决了野指针的问题。

iOS 内存管理的一点小问题的更多相关文章

  1. [转载]对iOS开发中内存管理的一点总结与理解

    对iOS开发中内存管理的一点总结与理解   做iOS开发也已经有两年的时间,觉得有必要沉下心去整理一些东西了,特别是一些基础的东西,虽然现在有ARC这种东西,但是我一直也没有去用过,个人觉得对内存操作 ...

  2. iOS内存管理编程指南

    iOS 内存管理 目录[-] 一:基本原则 二:成员变量的内存管理 三:容器对象与内存管理 四:稀缺资源的管理 五:AutoRelease 六:其他注意事项 iOS下内存管理的基本思想就是引用计数,通 ...

  3. iOS内存管理(objective-c)

    移动app开发中,由于移动设备内存的限制,内存管理是一个非常重要的话题.objective-c的内存管理,不仅是面试当中老生常谈的一个必问话题,也是日常项目开发中,特别需要重视的环节.对于笔者这种以j ...

  4. iOS内存管理布局及管理方案-理论篇

    苹果设备备受欢迎的背后离不开iOS优秀的内存管理机制,那iOS的内存布局及管理方案是怎样的呢?我们一起研究下. 内存管理分为五大块 栈区(stack):线性结构,内存连续,系统自己管理内存,程序运行记 ...

  5. iOS内存管理

    iOS内存管理的方式是引用计数机制.分为MRC(人式引用计数)和ARC(自动引用计数). 为什么要学习内存管理? 内存管理方式是引用计数机制,通过控制对象的引用计数来实现操作对象的功能.一个对象的生命 ...

  6. 【Bugly干货分享】iOS内存管理:从MRC到ARC实践

    Bugly 技术干货系列内容主要涉及移动开发方向,是由Bugly邀请腾讯内部各位技术大咖,通过日常工作经验的总结以及感悟撰写而成,内容均属原创,转载请标明出处. 对于iOS程序员来说,内存管理是入门的 ...

  7. iOS内存管理个人总结

    一.变量,本质代表一段可以操作的内存,她使用方式无非就是内存符号化+数据类型 1.保存变量有三个区域: 1>静态存储区 2>stack 3>heap 2.变量又根据声明的位置有两种称 ...

  8. IOS内存管理学习笔记

    内存管理作为iOS中非常重要的部分,每一个iOS开发者都应该深入了解iOS内存管理,最近在学习iOS中整理出了一些知识点,先从MRC开始说起. 1.当一个对象在创建之后它的引用计数器为1,当调用这个对 ...

  9. iOS内存管理(一)

    最近有时间,正好把iOS相关的基础知识好好的梳理了一下,记录一下内存相关方面的知识. 在理解内存管理之前我觉得先对堆区和栈区有一定的了解是非常有必要的. 栈区:就是由编译器自动管理内存分配,释放过程的 ...

随机推荐

  1. Access与SQL中的IsNull(),IS NULL的区别

    Access也有IsNull函数,但意义和参数却和T-SQL中的不同. 在T-SQL(也就是SQL Server所支持的SQL语言)中,IsNull的作用是把空值替代成指定的值.然而在Access中, ...

  2. 【435】NULL '\0' 0 等在 C 语言中的区别

    参考:C/C++语言中NULL.'\0’和0的区别 参考:空字符串.'\0'.0与NULL的区别以及数组清零的特点分析 在 C语言 中没有 空字符 这个东西 '',不过有 空字符串 "&qu ...

  3. ECMAScript 6复习<一>

    1.let和const命令: let不存在变量提升 暂时性死区 let在相同作用域内不允许重复声明 2.块级作用域: 3.全局对象的属性: ; window.a let b = ; window.b ...

  4. 【Leetcode_easy】892. Surface Area of 3D Shapes

    problem 892. Surface Area of 3D Shapes 题意:感觉不清楚立方体是如何堆积的,所以也不清楚立方体之间是如何combine的.. Essentially, compu ...

  5. js 次方 开方 对数

    次方 ,用Math.pow(值,次方数) 如: Math.pow(3,2);   3的平方 Math.Pow(2,3);   2的立方 开方Math.sqrt(值) 如: Math.sqrt(9);  ...

  6. 转录组组装软件stringtie

    StringTie是約翰·霍普金斯大學计算机生物中心开发的一款转录组组装软件,在组装转录本的完整度,精度和速度方面都较以往的cufflinks 有很大的提升,也是目前有参考基因组转录组主流的组装软件. ...

  7. 异常查错java.net.SocketException: Connection reset

    用httpclient访问后台接口报错java.net.SocketException: Software caused connection abort: recv failed,百度了一圈都说是由 ...

  8. 转:Cesium 和 Webpack

    原文地址:https://www.jianshu.com/p/85917bcc023f 注意:webpack 和 webpack-cli 的安装参考 https://www.cnblogs.com/m ...

  9. Linux shell 中 & && [] [[]] () [] 含义

    | 语法:command 1 | command 2 功能:把第一个命令 command 1 执行的结果作为 command 2 的输入传给 command 2 & & 放在启动参数后 ...

  10. Python18之函数定义及调用,注释

    一.函数定义 def 函数名(形参1,形参2...): 函数体 return 返回值         (可以返回任何东西,一个值,一个变量,或是另一个函数的返回值,如果函数没有返回值,可以省略retu ...