我一直以为现在都是自动内存管理了,还哪有什么内存泄漏啊。一检测才知道,不是我太相信Xcode了,就是我太单纯了。iOS开发中遇到的内存泄漏主要有几下几种:

(1)对象不能释放。使用Core Foundation对象的时候要特别注意,因为他还是MRC,需要自己释放对象。

(2)野指针。这儿比较危险,调用一个不属于你的对象,发生什么谁都不知道。

(3)空指针。OC这一点很好,如果你往一个空对象发送消息不会报错和崩溃。

  检测内存泄漏我一般用两种工具Analyze、Leaks。本文主要介绍我在开发中遇到的内存泄漏问题的原因和解决方案,至于怎么使用这两个检测工具可以看这篇文章,以后会不断更新。

Analyze 检测的问题

一、Memory error

1.Null passed to a callee that requires a non-null 1st parameter

调用的方法需要一个非空的参数。

e.g:

UIImage *image = [UIImage imageWithData:data];

原因:把二进制数据流转换成图片,如果data为nil,就不应该调用这个方法。不过不会崩溃,亲测。

修改:

if(data) {
UIImage *image = [UIImage imageWithData:data];
}

二、Merrory(Core Foundation/Objective-C)

1.Potential leak of an object stored into ‘cs’

调用了让某个对象引用计数加1的函数,但没有调用相应让其引用计数减1的函数。

e.g:

CGColorSpaceRef cs = CGColorSpaceCreateDeviceGray();
CGContextRef bitmapRef = CGBitmapContextCreate(nil, width, height, 8, 0, cs, (CGBitmapInfo)kCGImageAlphaNone);

原因:Core Foundation是一组C语言框架,不是ARC的,需要手动释放内存,这个比较坑。

修改:

CGColorSpaceRef cs = CGColorSpaceCreateDeviceGray();
CGContextRef bitmapRef = CGBitmapContextCreate(nil, width, height, 8, 0, cs, (CGBitmapInfo)kCGImageAlphaNone);
CGColorSpaceRelease(cs);

(三)Dead store

1.Value stored to ‘payResoult’ during its initialization is never read.

存储在'payResoult'里的值从未被读取过。

e.g:

 NSString *str = @"Dead store ";
str = @"Dead store never used";

原因:你往内存里申请了空间,存了值,但是你存的值从来没有用过,所以白白消耗了内存。

处理:既然没用就删去吧。

(四)Coding Conventions(Apple)

Method accepting NSError** should have a non-void value to indicatie whether or not an error occurred;

e.g:

- (void)doDecodeMultiple:(ZXBinaryBitmap *)image hints:(ZXDecodeHints *)hints results:(NSMutableArray *)results xOffset:(int)xOffset yOffset:(int)yOffset currentDepth:(int)currentDepth error:(NSError **)error;

原因: (NSError**)error -> error 的地址(&error)。当error为nil的时候,&error也就没有意义了,因为不可能一直有错误。

处理:暂时还不知道,有没有哪位大神可以告诉啊

Leaks 检测的问题

一.AFNetworking

e.g

AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; manager.requestSerializer = [AFJSONRequestSerializer serializer];//请求
manager.responseSerializer = [AFHTTPResponseSerializer serializer];//响应
manager.requestSerializer.timeoutInterval = 5.0f; [manager POST:strURL parameters:parameters progress:^(NSProgress * _Nonnull uploadProgress) { } success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { responseObject = [FHTool ResultDic:responseObject];
if (responseObject) {
if (networkInfoBLock) {
networkInfoBLock(NetworkReturnCodeSuccess,responseObject);
} }else {
if (networkInfoBLock) {
networkInfoBLock(NetworkReturnCodeFailed,nil);
}
} } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"strURL = %@\n error = %@",strURL,error.localizedDescription);
if (networkInfoBLock) {
networkInfoBLock(NetworkReturnCodeFailed,@{@"error":error.localizedDescription});
} }];

原因: 每次请求数据,都会创建一个AFHTTPSessionManager对象,而在ARC模式下,他们不能够被释放。

解决:把数据请求类封装成一个单例,只创建一个AFHTTPSessionManager对象。

NetworkManager.h

#import <Foundation/Foundation.h>
#import <AFNetworking.h> typedef void (^NetworkInfoBLock)(NetworkReturnCode networkReturnCode ,NSDictionary *responseObject); @interface NetworkManager : NSObject @property (nonatomic, strong) AFHTTPSessionManager *manager; + (instancetype) shareNetworkManager;
// post
- (void)postNetworkByURL: (NSString *)strURL AndParameter: (id)parameters AndNetworkSuccessBLock:
(NetworkInfoBLock)networkInfoBLock; @end

NetworkManager.m

#import "NetworkManager.h"

@implementation  NetworkManager

+ (instancetype) shareNetworkManager {

    static  NetworkManager * networkManager = nil;
static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ networkManager = [[NetworkManager alloc] init]; AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
manager.requestSerializer = [AFJSONRequestSerializer serializer];//请求
manager.responseSerializer = [AFHTTPResponseSerializer serializer];//响应
manager.requestSerializer.timeoutInterval = 5.0f; networkManager.manager = manager;
}); return networkManager;
}
- (void)postNetworkByURL: (NSString *)strURL AndParameter: (id)parameters AndNetworkSuccessBLock:
(NetworkInfoBLock)networkInfoBLock { [self.manager POST:strURL parameters:parameters progress:^(NSProgress * _Nonnull uploadProgress) { } success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { responseObject = [FHTool ResultDic:responseObject];
if (responseObject) {
if (networkInfoBLock) {
networkInfoBLock(NetworkReturnCodeSuccess,responseObject);
} }else {
if (networkInfoBLock) {
networkInfoBLock(NetworkReturnCodeFailed,nil);
}
} } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { NSLog(@"strURL = %@\n error = %@",strURL,error.localizedDescription);
if (networkInfoBLock) {
networkInfoBLock(NetworkReturnCodeFailed,@{@"error":error.localizedDescription});
} }]; }
@end

iOS 内存泄漏的更多相关文章

  1. 讲述Sagit.Framework解决:双向引用导致的IOS内存泄漏(中)- IOS不为人知的Bug

    前言: 话说昨晚还是前晚,写了一篇:讲述Sagit.Framework解决:双向引用导致的IOS内存泄漏(上) 文章写到最后时,多了很多莫名奇妙的问题!!! 为了解决了这些莫名奇妙的问题,我又战斗了2 ...

  2. 讲述Sagit.Framework解决:双向引用导致的IOS内存泄漏(下)- block中任性用self

    前言: 在处理完框架内存泄漏的问题后,见上篇:讲述Sagit.Framework解决:双向引用导致的IOS内存泄漏(中)- IOS不为人知的Bug 发现业务代码有一个地方的内存没释放,原因很也简单: ...

  3. iOS内存泄漏自动检测工具PLeakSniffer

    新款objective-C内存泄漏自动检测工具 PLeakSniffer , GitHub地址 (https://github.com/music4kid/PLeakSniffer). 背景 前些天读 ...

  4. 使用Xcode和Instruments调试解决iOS内存泄漏

    尽管iOS 5.0加入版本号之后ARC机制,由于相互引用关系是复杂的.内存泄漏可能仍然存在.于是,懂原理是非常重要的. 这里讲述在没有ARC的情况下,怎样使用Instruments来查找程序中的内存泄 ...

  5. 经验之道:最有效的iOS内存泄漏检测

    版权声明:本文由胡涛原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/125 来源:腾云阁 https://www.qclou ...

  6. iOS 内存泄漏监测自动化

    在 Android 上,Square 这家公司提供了非常有名的工具: leakcanary ,来帮助开发者们在日常开发过程中就能够发现内存泄漏.但在 iOS 上呢?在 Google 的时候,我发现了两 ...

  7. 一起来看看IOS内存泄漏的一个问题

    很多iOS开发的朋友都是比较关心内存泄漏的问题,在实际的开发工作中首先我们需要知道程序有没有内存泄露,然后定位到底是哪行代码出现内存泄露了,这样才能将其修复.最简单的方法当然是借助于专业的检测工具,比 ...

  8. IOS内存泄漏

    1. . - (void)viewDidLoad { [superviewDidLoad]; self.view.frame=CGRectMake(, , , ); NSArray *title1=[ ...

  9. iOS 内存泄漏排查以及处理

    使用Xcode7的Instruments检测解决iOS内存泄露   文/笨笨的糯糯(简书作者)原文链接:http://www.jianshu.com/p/0837331875f0作为一名iOS开发攻城 ...

随机推荐

  1. python3 - 通过BeautifulSoup 4抓取百度百科人物相关链接

    导入需要的模块 需要安装BeautifulSoup from urllib.request import urlopen, HTTPError, URLError from bs4 import Be ...

  2. phpcms基础知识和配置

    一.设置界面 1.站点设置:相当于服务器上的站点 (1)站点修改:“关键词”和“描述”的修改,便于网络优化和搜索引擎对本网站的搜索. (2)模板的修改,可以自己加模板,引用自己模板 2.基本设置:所有 ...

  3. Ffmpeg 视频教程

    最近一段时间找时间录制了一些Ffmpeg视频教程,还有录制完毕,会持续更新,内容会包含Ffmeg保存文件,网络流转发, 编码,解码,播放器制作,以及服务端搭建等等,适合初学者,有需要的朋友的可以关注: ...

  4. spring入门--Spring框架底层原理

    上一篇的博客,我们可以看出来,spring可以维护各个bean (对象),并向其中注入属性值.那么,如果们要把一个对象的引用注入另外一个对象呢?应该怎么处理呢? 我们知道,对于对象中的属性来说,我们注 ...

  5. C# 6 与 .NET Core 1.0 高级编程 - 40 ASP.NET Core(下)

    译文,个人原创,转载请注明出处(C# 6 与 .NET Core 1.0 高级编程 - 40 章  ASP.NET Core(下)),不对的地方欢迎指出与交流. 章节出自<Professiona ...

  6. 远程线程注入方法CreateRemoteThread

    最近在整理学习Windows注入方面的知识,这个远程注入前面早写过,现在看看人家博客的理解整理,整理, 需要源码的可以到我的github上下载. 链接是  https://github.com/Ars ...

  7. 【UWP】拖拽列表项的排序功能实现

    在一些允许用户自定义栏目顺序的app(如:凤凰新闻.网易云音乐等),我们可以方便地拖拽列表项来完成列表的重新排序,进而完成对栏目顺序的重排.这个功能很人性化,而实现起来其实很简单(甚至都不用写什么后台 ...

  8. loadrunner入门篇-Controller控制器

    Controller组件是LR的控制中心,主要包括场景设计和场景执行两部分.在VuGen中编辑完脚本并将脚本加载到Controller组件中,即开始对脚本运行时的场景进行设计,当场景设计完成后,即可执 ...

  9. 对java数组的一些理解

    刚开始学习Java的时候一直搞不清除获取数组的长度是用length()还是length,现在不妨来深入了解一下数组的真实面目. 我们不妨来看一下数组的源码,诶,数组的类名叫什么?我们声明一个int数组 ...

  10. Yii前台后台登录混淆问题

    我们在用yii开发项目时候,如果前后台使用modules实现 那么 做登录时候用户名就会出现前后台登录混淆的事情 于是可以在adminModule.php文件中做个设置 public function ...