(转)(译)怎样在xcode里面使用Memory Leaks和Instruments教程

2024-03-15 13:32

本文主要是介绍(转)(译)怎样在xcode里面使用Memory Leaks和Instruments教程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文链接地址:http://www.raywenderlich.com/2696/how-to-debug-memory-leaks-with-xcode-and-instruments-tutorial

  免责申明(必读!):本博客提供的所有教程的翻译原稿均来自于互联网,仅供学习交流之用,切勿进行商业传播。同时,转载时不要移除本申明。如产生任何纠纷,均与本博客所有人、发表该翻译稿之人无任何关系。谢谢合作!


  本文由yy翻译。Powered by YY!:)

教程截图:

  作为一名无证程序员,无论你多么精通Objective-C的内存管理,随着时间的推移,你也不可避免的犯内存相关的错误。但通常因为代码量太大,以至于你不可能一行一行的去排除(等你解决完,你设计的动车早相撞了!)

  幸运的是,苹果已经提供了一些好的方式来帮助你找到应用程序中内存相关的问题。有时,这些工具可能吓到初学者,但它们实际上相当有用并易于掌握!

  这就是本教程说要介绍的.你会亲手使用内存工具在XCode环境下很轻松的检测内存问题。

  这篇教程是建立在你非常熟悉Objective-C内存管理的基础上。如果你还在这个问题上找不着北,你可能需要学习内存管理其他教程。

第一步

  在这一节中,我们的目的是在一个例子应用程序中检查、解决任何内存泄漏问题,以演示常见的内存相关错误处理。开始,下载一个应用程序示例。我已经将教程和示例工程文件放在一起了。

  在XCode中打开工程并运行。你会看到tableview中包含了一个寿司列表。试着选择几行,然后——轰!你看到可怕的EXC_BAD_ACCESS错误,编译器拿它完全没有办法。

  因为xcode完全没指出出问题的地方,所以这种情况通常令许多开发者感到郁闷。当你遇到了一个EXC_BAD_ACCESS错误,我通常会给开发者几个建议:

  1.在可执行选项中设置NSZombieEnabled参数,这有时会帮缩小问题的范围;

  2.运行apple的内存检测工具, Leaks ,以便寻找内存问题;

  3设定一个断点,单步运行代码,直到你找到引起崩溃的位置;

  4.注释代码,直到不崩溃为止,然后再从后往前查找错误;

  现在让我们从第一条开始实验

  # 1 - NSZombieEnabled参数

一大波僵尸正在靠近!!!!

  不幸的是,NSZombieEnabled选项对于崩溃毫无办法,所以你完全可以放弃抵抗。

  当你试图使用一个已经被销毁的对象,NSZombieEnabled会标志一个警告,所以NSZombieEnabled只是一个flag。这是一个良好的开端,因为大多数崩溃的原因都是使用了已经销毁的对象。

  按照以下设置:在XCode中展开Executables->双击PropMemFun->选择Arguments选项卡->Variables to be set in the environment”点击加号按钮。把变量名值设置成NSZombieEnabled,把值设置成YES,如下图:(xcode4在左上角,edit schema里面)

  重新运行app,随便操作下使程序崩溃。 查看下console log你就会看到如下信息:

2011-02-03 12:07:44.778 PropMemFun[27224:207] ***
-[CFString respondsToSelector:]: message sent to deallocated instance ...

  这个程序将在很精确的一行暂停。崩溃后,你可以通过选定第一个区域,回溯找出导致崩溃的准确行数。比如现在这个示例就崩溃在:tableView:didSelectRowAtIndexPath

  不管你信不信,反正找出了出问题的那行。导致崩溃的问题就是向已经销毁的string发送了一个消息。这一行用了两个string _lastSushiSelectedsushiString.

  因为这个string是由stringWithFormat初始化,所以看起来程序是没有问题了,因为stringWithFormat的返回值是自动释放的,所以在下次使用前应该是安全的。但是_lastSushiSelected的安全性如何呢?

  虽然_lastSushiSelected是在sushiString执行到最后才赋值的。但是sushiString是自动释放的,所以有些时候sushiString被释放了,内存也被销毁。但是紧接着_lastSushiSelected 仍然有可能指向被销毁的内存!这就解释了崩溃原因:向已经销毁的内存发送消息导致崩溃。

  我们只需保留_lastSushiSelected就可以解决这个问题,把最后一行改成下面的样子:

_lastSushiSelected = [sushiString retain];

  再次运行程序,你会发现程序已经畅通无阻了。

编译,分析和总结

  至少,我们有一个不崩溃的应用程序——这是一个好的开始。但接下来,我们需要开始确保没有任何内存泄漏。

  有一种简单的方法可以初步确认你的程序在初始化中是否有任何内存泄漏或其他问题--使用内置编译和分析功能(built-in Build and Analyze)。

  这将使XCode执行你的代码和自动检测任何错误并警告你任何潜在的问题。它并不会找出所有的问题,但用这个方法找出的错误无疑是一个既快速又简单的方法。

  试一试通过选择Build/Build and Analyze。你应该看到,它检测到一个内存泄漏,你可以看到如下:

  消息显示,alertView”有一个潜在的内存泄漏。如果你看看这一行,你就会发现所有的UIAlertView创造是有着alloc /init (返回一个对象引用数1),却从来没有真正地释放!有几种方法可以解决这个问题,但其中一个方法就是在[alertView show]下面加上一行:

[alertView release];

  再次 Build/Build and Analyze,你会发现已经找不出任何内存问题了。

泄漏和管道 

  不幸的是,你不能依靠Build/Build and Analyze找出一切问题。有一个强大的自动化工具来帮助你检查程序是否有内存泄漏– the Leaks Instrument

  让我们试试看。选择Run/Run -> Performance -> Tool/Leaks,再选择table view中的几行。也可以上下滚动table view,从table view顶端到底部。基于前面的经验,你就应该开始看出一些蓝色的标签出现在泄漏的内存上。

  点击停止按钮,然后去工具栏中点击“Leaked Blocks”让他变成“Call Tree”。在面板左下角,点击“Invert Call Tree”、“Hide System Libraries”。你将会看到这个工具发现两个不同的函数存在内存泄漏,你可以看到如下:

  如果你双击一个函数的名字,它会带你直接到存在内存泄露的这行代码。这可以给你一个很好的错误位置提示,如果你查看代码并加以思考,你应该能够找出问题所在并解决它。

  所以,为什么不看看代码,并且看看你是否能找出问题所在并修正吗?一旦你作出修改,并且能够无错误提示的跑Leaks。如果通过,表示你完成了

  …

  …waiting…

  …

  …waiitng…

  …

  …waiting…

  …!

 

  你已经搞定,不管你信不信,反正我是信了。

解释一下

  tableView:didSelectRowAtIndexPath

  Leaks 告诉我们,这个问题的原因是字符串sushiString创造和存储过程中引起的内存泄漏。所以让我们一步一步的分析一下原因:

  1.sushiString被创建时,调用stringWithFormat 返回一个对象数值1并且发送autorelease消息

  2.在方法的最后一行,你在sushiString加入retain(retain数值增加到2)并将其存储到_lastSushiSelected

  3.后来,autorelease生效,retain数递减为1

  4.下一个tableView:didSelectRowAtIndexPath方法被调用,你重写_lastSushiSelected变量的一个指针指向一个新的字符串,- - - - -如果没有释放旧的所以那个老字符串并没有被释放仍然存在。

  一个解决办法是增加下面一行在初始化lastSushiSelected sushiString之前:

[_lastSushiSelected release];
 

  tableView:cellForRowAtIndexPath

  就像在前面的方法,创建和存入名为sushiString的变量引起内存泄漏。以下是引起问题的分析:

  1.一个新的字符串被alloc/init方法创建。

  2.返回一个对象引用数 1.

  3.然而,这个计数从来不减少,所以有一个内存泄漏!

  这可以通过三种方式中的一种解决:

  1.设置textLable为一个字符串后在sushiString中调用release方法。

  2.alloc/init方法初始化完毕后在sushiString中调用autorelease

  3.stringWithFormat代替alloc/init方法,返回一个已经标志为自动释放的字符串。

  验证 leaks

  修正前面提到的问题,再次运行leaks,你会得到一个没有任何内存泄漏的app

接下来该干什么?

  这个链接可以下载到一个已经解决上述问题的工程文件。

  最重要的是,你必须亲自实践使用NSZombieEnabledBuild and Analyze,和Leaks Instrument工具来找到内存泄漏。你应该能够很快把这项技术运用到你的工程中。

  如果你有更好的方法,可以在下面评论,我也积极采纳大家的建议。

这篇关于(转)(译)怎样在xcode里面使用Memory Leaks和Instruments教程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/812147

相关文章

一文详解如何使用Java获取PDF页面信息

《一文详解如何使用Java获取PDF页面信息》了解PDF页面属性是我们在处理文档、内容提取、打印设置或页面重组等任务时不可或缺的一环,下面我们就来看看如何使用Java语言获取这些信息吧... 目录引言一、安装和引入PDF处理库引入依赖二、获取 PDF 页数三、获取页面尺寸(宽高)四、获取页面旋转角度五、判断

Windows环境下解决Matplotlib中文字体显示问题的详细教程

《Windows环境下解决Matplotlib中文字体显示问题的详细教程》本文详细介绍了在Windows下解决Matplotlib中文显示问题的方法,包括安装字体、更新缓存、配置文件设置及编码調整,并... 目录引言问题分析解决方案详解1. 检查系统已安装字体2. 手动添加中文字体(以SimHei为例)步骤

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命

Java JDK1.8 安装和环境配置教程详解

《JavaJDK1.8安装和环境配置教程详解》文章简要介绍了JDK1.8的安装流程,包括官网下载对应系统版本、安装时选择非系统盘路径、配置JAVA_HOME、CLASSPATH和Path环境变量,... 目录1.下载JDK2.安装JDK3.配置环境变量4.检验JDK官网下载地址:Java Downloads

使用Python删除Excel中的行列和单元格示例详解

《使用Python删除Excel中的行列和单元格示例详解》在处理Excel数据时,删除不需要的行、列或单元格是一项常见且必要的操作,本文将使用Python脚本实现对Excel表格的高效自动化处理,感兴... 目录开发环境准备使用 python 删除 Excphpel 表格中的行删除特定行删除空白行删除含指定

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

prometheus如何使用pushgateway监控网路丢包

《prometheus如何使用pushgateway监控网路丢包》:本文主要介绍prometheus如何使用pushgateway监控网路丢包问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录监控网路丢包脚本数据图表总结监控网路丢包脚本[root@gtcq-gt-monitor-prome

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数

SpringBoot中如何使用Assert进行断言校验

《SpringBoot中如何使用Assert进行断言校验》Java提供了内置的assert机制,而Spring框架也提供了更强大的Assert工具类来帮助开发者进行参数校验和状态检查,下... 目录前言一、Java 原生assert简介1.1 使用方式1.2 示例代码1.3 优缺点分析二、Spring Fr