1.前言
本来以为在改成ARC以后,不再需要考虑内存问题了,可是在实践中还是发现有一些内存问题需要注意,今天我不谈block的循环引用的问题,主要说说一些对象、数组不内存得不到释放的情况.
2.数组内存得不到释放的情况
//组织字典数据 - (NSMutableDictionary *)setupDicData{ NSMutableDictionary *dict = [NSMutableDictionary dictionary]; for (int i = 0; i <= 30; i++) { [dict setObject:[self setupArrayData] forKey:[NSString stringWithFormat:@"%d%@",i,@"class"]]; } return dict; } //组织数组数据 - (NSMutableArray *)setupArrayData{ NSMutableArray *marry = [NSMutableArray array]; for (int i = 0; i<=30; i++) { NSString *s = [NSString stringWithFormat:@"%@",@"data-test"]; [marry addObject:s]; } return marry; }
运行+——
- (void)viewDidLoad { [super viewDidLoad]; while (true) { //30.0定时执行 [NSThread sleepForTimeInterval:30.0]; NSDictionary *dict = [self setupDicData]; NSLog(@"%@",dict); //每次数据内存都得不到释放 } }
//按上代码传递数组执行,每次数组、对象内存都得不到释放。如图:
内存会无线的往上增加,直至崩溃。
2.是什么原因导致这种内存得不到释放的?
主要是你在iOS里使用 while (true) {} 无线循环时,
iOS ARC默认认为你这个方法永远没有执行完,所以不会去主动释放你方法里的对象,这一点和JAVA不一样,
所以很多JAVA开发者转iOS后习惯性的使用while(true){}
导致项目里存在这种内存隐患,导致内存无限增加。
3.如何解决这种数组传递内存得不到释放的情况?
解决方法一:
3.1.最简单最直接在ARC的环境下使用 @autoreleasepool {}
//@autoreleasepool {}的作用是在每次循环一次,都会把内存主动释放掉 - (void)viewDidLoad { [super viewDidLoad]; while (true) { @autoreleasepool { //30.0定时执行 [NSThread sleepForTimeInterval:30.0]; NSDictionary *dict = [self setupDicData]; NSLog(@"%@",dict); //每次数据内存都得不到释放 } } }
内存图,我们发现很稳定,每次都会主动将内存释放
解决方法二:
3.2.使用NSTimer来做数组传递的无限循环,ARC会自动帮你释放内存
- (void)usingDatadosomething{ //30.0定时执行 [NSThread sleepForTimeInterval:0.10]; NSDictionary *dict = [self setupDicData]; NSLog(@"%@",dict); //每次数据内存都得不到释放 } - (void)viewDidLoad { [super viewDidLoad]; [NSTimer scheduledTimerWithTimeInterval:30.0 target:self selector:@selector(usingDatadosomething) userInfo:self repeats:YES]; [[NSRunLoop currentRunLoop] run]; }
内存图如下
解决方法三:
3.3.使用block封装数组传递,最后做block的释放,ARC会自动帮你释放内存
block使用较为频繁,不在本文章阐述,下次会专门写一篇block的博客。
谢谢大家,还有什么疑问可以评论中提出,我有时间会耐心回复!