写程序遇到 Bug 并不可怕,大部分的问题,经过简单的 Log 或者 代码分析并不难找到缘由所在。可是在 Objective-C 编程中遇到 EXC_BAD_ACCESS 问题的时候,经过简单常规的手段很难发现问题。这篇文章,给你们介绍一个经常使用的查找 EXC_BAD_ACCESS 问题根源的方法。编程
首先说一下 EXC_BAD_ACCESS 这个错误,能够这么说,90%的错误来源在于对一个已经释放的对象进行release操做。spa
Objective-C 这段代码有三个致命问题:一、内存泄露;二、错误释放;三、形成 EXC_BAD_ACCESS 错误。对象
1, NSString* s = [[NSString alloc]initWithString:@”This is a test string”]; 建立了一个 NSString Object,随后的 s = [s substringFromIndex:[s rangeOfString:@"a"].location]; 执行后,致使建立的对象引用消失,直接形成内存泄露。内存
2,错误释放。[s release]; 这个问题,缘由之一是一个逻辑错误,觉得 s 仍是咱们最初建立的那个 NSString 对象。第二是由于从 substringFromIndex:(NSUInteger i) 这个方法返回的 NSString 对象,并不须要咱们来释放,它实际上是一个被 substringFromIndex 方法标记为 autorelease 的对象。若是咱们强行的释放了它,那么会形成 EXC_BAD_ACCESS 问题。string
3, EXC_BAD_ACCESS。因为 s 指向的 NSString 对象被标记为 autorelease, 则在 NSAutoreleasePool 中已有记录。可是因为咱们在前面错误的释放了该对象,则当 [pool drain] 的时候,NSAutoreleasePool 又一次的对它记录的 s 对象调用了 release 方法,但这个时候 s 已经被释放不复存在,则直接致使了 EXC_BAD_ACCESS问题。it
查看更多的Console信息io
工做区->Excuteables->双击其分组下的文件->Arguments设置运行参数table
1: 为工程运行时加入 NSZombieEnabled 环境变量,则在 EXC_BAD_ACCESS 发生时,XCode 的 Console 会打印出问题描述。class
首先双击 XCode 工程中,Executables 下的 可执行模组,test
2:加入 MallocStackLogging 来启用malloc记录
作以下设置:
Project -> Edit active executable ->Argument
添加以下四个参数
NSDebugEnabled
NSZombieEnabled
MallocStackLogging
MallocStackLoggingNoCompact