优化内存泄漏的几点分析

很久很久没有写过文章了,割的时间有点长了,长的都让我忘却了学习.这里用小锤锤锤一下自己,咋就不知道学习啦!

好了废话不多说:导致内存泄漏的几点原因(不全面,请在回复区补充)

1.关于cf框架的使用

大家都知道在使用Core Foundation的API时要时刻注意自己需要手动释放创建的内存,凡是看到create,copy,mutablecopy,alloc等都要调用CFRelease

举例如下:

CFUUIDRef uuid = CFUUIDCreate(NULL);

appUID = (NSString *) CFUUIDCreateString(NULL, uuid);

CFRelease(uuid);


2.关于block的循环引用

这个东西已经被说过很多次了,导致循环引用的原因也是很容易分析的:实例对象引用了block,在block内部访问了实例对象(包括该对象的实例变量)就会产生循环引用的问题

举例如下:

//情况一

- (void)case1 {

NSLog(@"case 1 Click");

dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

self.name = @"case 1";

});

}

这种情况不会造成内存泄漏,原因很简单,他并不是一个闭环,虽然在block内部引用了self,但是self并没有持有block,这就是常见的系统级别的block不用使用weakself,和strongself的原因,如uiview的动画block,gcd等

//情况二

- (void)case2 {

NSLog(@"case 2 Click");

__weaktypeof(self) weakSelf = self;

[self.teacher requestData:^(NSData *data) {

typeof(weakSelf) strongSelf = weakSelf;

strongSelf.name = @"case 2";

}];

}

第二种情况就是我们常见的容易出现循环引用的地方,self持有了block,block持有了self,形成了一个完美的闭环,根本无法释放,要想释放内存就必须保证一端为弱引用.但是在block内部为了防止self提前释放,又转成了strongself,也就是强引用保证在block内部self永远存在!

//情况三

- (void)case3 {

NSLog(@"case 3 Click");

[self.teacher requestData:^(NSData *data) {

self.name = @"case 3";

}];

}

内存泄漏,循环引用,原因就是相互强引用导致

//情况四

- (void)case4 {

NSLog(@"case 4 Click");

[self.teacher requestData:^(NSData *data) {

self.name = @"case 4";

self.teacher = nil;

}];

}

不存在内存泄漏了和循环引用,在block执行结束后主动释放了block的持有者,最终或导致self的dealloc执行.所以并不会内存泄漏,常常会看见有些大神在解决block的问题时在block执行结束后手动释放掉block,原理就是如此.

//情况五

- (void)case5 {

NSLog(@"case 5 Click");

Teacher *t = [[Teacher alloc] init];

[t requestData:^(NSData *data) {

self.name = @"case 5";

}];

}

不会造成循环引用,不会内存泄漏,因为是局部变量持有block,在这个方法的大括号内部有效,出了大括号局部变量就会被释放.所以不存在内存问题.

//情况六

- (void)case6 {

NSLog(@"case 6 Click");

[self.teacher callCase6BlackEvent];

self.teacher.case6Block = ^(NSData *data) {

self.name = @"case 6";

//下面两句代码任选其一

self.teacher = nil;

//        self.teacher.case6Block = nil;

};

}

self.teacher = nil; , self.teacher.case6Block = nil;这两句代码任选其一即可解决内存泄漏,因为一端被置为nil,就会打破循环引用问题.

3.实际项目中可能会遇到AFNetworking的内存泄漏问题

为什么会有这样的问题?

在实例化 AFHTTPSessionManager中如果进行多次调用,就会导致内存泄漏例如在项目中同事发起多个请求,创建多个manager就会有这样的问题.

[[AFHTTPSessionManager alloc] initWithSessionConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]];

解决办法:采用单例的方式创建,或者是继承AFHTTPSessionManager并将该类做成一个单例类

4.在for循环较大数据时,不断创建局部变量导致的内存泄漏问题

for (NSDictionary *dic in regionList) {

@autoreleasepool {

RegionList *model = [RegionList yy_modelWithDictionary:dic];

NSString *city;

if (model.city.length == 0) {

city = [NSString stringWithFormat:@"%@",model.region];

model.cityName = city;

}

else {

city = [NSString stringWithFormat:@"%@·%@",model.city,model.region];

model.cityName = city;

}

if (city.length > 0) {

model.cityPinyin = [city pinYin];

NSString *str = [model.cityPinyin uppercaseString];

model.firstChar = [str substringToIndex:1];

}

[regionListModel addObject:model];

}

}

举例:

在项目中可能会遇到从服务端获取城市列表,全国大概有三百多个市,每个市估算有10个区那么就会有3000多条数据.

利用yymodel解析从后台拿到的数据时是不是需要在for循环中创建3000多个局部变量.3000多个局部变量占用的内存将会是爆发式的增长.到底有多恐怖还请自己测一下,可能3000看不出多少效果可以给大点数字试试.

结论:

这个for循环里如果不使用@autoreleasepool,那临时变量内存可能是爆发式的,但是使用了@autoreleasepool,在每个@autoreleasepool结束时,里面的临时变量都会回收,内存使用更加合理


5.mrc的基础

自己生成的对象,自己持有 (alloc ,new,create)

不是自己生成的对象,自己也能持有(retain,copy,mutablecopy)

谁持有,谁释放不持有,不能释放不再需要时,主动释放 (release)


6,使用工具调试内存泄漏问题

存在内存泄漏

如何查看找到源代码位置


选择calltreses

选择调用树


关闭掉系统方法的调用

这样你就能看到你带吗内存泄漏的问题所在了.

内存泄漏代码

可能看到的代码并不存在内存泄漏问题,需要通过上下文分析内存可能泄漏的原因.


本人联系方式:qq:513961360

email:513961360@qq.com

也可以加我们的qq群希望能与朋友们一起聊天和学习.群里还有很多iOS开发者,帮助我们解决问题,并且同时学习.

qq群号:580284575

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容

  • 一:block内部可能存在的self的集中使用情况 (1)什么时候在 block 里面用 self,不需要使用 w...
    雷鸣1010阅读 1,220评论 0 1
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,587评论 18 399
  • Leaks 苹果官方有关于内存分析说明,在文档中把内存泄漏情况分为4中: Overall Memory Use. ...
    iven_zf阅读 2,920评论 1 11
  • iOS开发中, 之前一直使用swift, 因此对于Objective-C的内存管理机制长期处于混乱的一知半解状态....
    icetime17阅读 838评论 1 8
  • 在一个微信群里,别人整理自己的文字。我真的开心,我要记下来,自我欣赏,嘿嘿 1.自己有了,才愿意给出,无法给出自己...
    典奇nvc阅读 316评论 2 2