iOS开发 一些常见的内存泄露和崩溃

项目(ARC)开发过程中,难免遇到内存泄漏和崩溃,特在这整理一下。

(如果本文中有讲述不对或者不准确的地方欢迎大家提出来)

一、内存泄漏

1、EXC_BAD_ACCESS / KERN_INVALID_ADDRESS

内存泄漏

公司的项目接入了三方崩溃报告,最近出现了EXC_BAD_ACCESS / KERN_INVALID_ADDRESS这样的错误,崩溃报告堆栈信息一大堆,看的头晕。
How to fix it?
这个奔溃常见于block的循环引用所造成的内存泄漏

1
2
3
4
5
6
//一个简单的例子
SecondViewController *secondVC = [[SecondViewController alloc] init];
__weak __typeof(&*self)weakSelf = self;
[secondVC setChangeLabel:^(NSString *text) {
       weakSelf.text = text;
}];

我们定义一个wself变量并加上__weak修饰符,在Block中,所有需要self的地方都用wself来替代。这样就不会增加引用计数,所以block持有self对象也就不会造成循环引用,从而造成内存泄漏。


2、Value stored to ‘dic’ during its initialization is never read


内存泄漏

这个是在测试项目内存泄漏的时候遇到的 (Product->Analyze)

1
2
3
//内存泄漏代码
NSDictory *dic = [[NSDictory alloc] init];
dic = {@"0",@"1",@"2"};

这么写真是太逗了,像什么NSArrayNSString等等都有可能遇到这种低级错误

1
2
//这么写就好了啊
NSDictory *dic = {@"0",@"1",@"2"};


3、Potential leak of an object stored into ‘string’


内存泄漏

这个同样是在测试项目内存泄漏的时候遇到的 (Product->Analyze)

1
2
3
4
5
6
7
8
//why crash
+ (NSString *)uuidString {

CFUUIDRef theUniqueString = CFUUIDCreate(NULL);
CFStringRef string = CFUUIDCreateString(NULL, theUniqueString);
CFRelease(theUniqueString);
return(__bridgeNSString *)string;
}

这个是没有release对象造成的。

1
2
3
4
5
6
7
8
9
10
//fix it
+ (NSString*)uuidString {

CFUUIDReftheUniqueString =CFUUIDCreate(NULL);
CFStringRefstring =CFUUIDCreateString(NULL, theUniqueString);
NSString*tmpString = (__bridgeNSString*)string;
CFRelease(theUniqueString);
CFRelease(string);
returntmpString;
}

像什么subImageRef, CGGradientRef,CTFrameRef等等都有可能遇到这种错误。


二、崩溃

1、

僵尸错误

我们可以打开Product->Scheme->Edit Scheme->Run->DiagnosticsEnable Zombie Objects勾选上,然后打全局断点进行测试,如果复现了这个崩溃,控制台会输出错误信息,例如-[XXXXX getObjectAt:]: message sent to deallocated instance 0x11562a300

2、

插入数组的对象不能为空

我们可以在添加到数组之前做一下判断

1
2
3
4
5
if (nameStr == nil || [nameStr isKindOfClass:[NSNull class]])  {
[nameArr addObject:@"匿名用户"];
}else {
[nameArr addObject:nameStr];
}

3、

空数组越界
可变数组越界

我们可以自己写一个方法代替系统的获取数组中对象的方法objectAtIndex:

1
2
3
4
5
6
7
8
9
10
11
12
- (id)objectAtIndexCheck:(NSUInteger)index{
//越界
if (index >= [self count]) {
return nil;
}
id value = [self objectAtIndex:index];
//对象为空
if (value == [NSNull null]) {
return nil;
}
return value;
}

4、

崩溃信息

1、使用标准的初始化方法:
NSDictionary *dictionary =[[NSDictionaryalloc] initWithObjectsAndKeys:value1,@”key1”,value2,@”key2”, value3 ,@”value3”,nil];

2、使用ios6.0以后新支持的初始化方法:
NSDictionary *dictionary =@{@”key1” : value1,@”key2” : value2,@”key3” : value3};

现在我们对value1 value2 value3进行赋值,并把value2设为nil指针:
NSString value1 =@”value1”;NSString value2 =nil;NSString *value3 =@”value3”;

这时如果使用第二种初始化方法,运行程序会发现崩溃,日志如下:

1
2
3
4
***
Terminating app due to uncaught exception 'NSInvalidArgumentException',
reason: '***
-[__NSPlaceholderDictionary initWithObjects:forKeys:count:]: attempt to insert nil object from objects[1]'

也就是说使用这种初始化方法的时候必须保证key跟value都不为nil,因此我们需要在初始化之前对其进行判断,如果为nil就不加入字典。但是如果有需求让value必须为空的时候,可以将value赋值为[NSNull null]
这样就可以成功插入字典

5、

崩溃信息

[cell setColumnTitle:[homeData objectAtIndex:row]];
这里其实是要传NSString类型,而其实返回的是NSArray类型,这个错误不会立即出发,而是过一段时间再出发,所以不好定位。



先写这么多,大家遇到的问题欢迎补充。

O(∩_∩)O