iOS 内存泄漏
我一直以为现在都是自动内存管理了,还哪有什么内存泄漏啊。一检测才知道,不是我太相信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 内存泄漏的更多相关文章
- 讲述Sagit.Framework解决:双向引用导致的IOS内存泄漏(中)- IOS不为人知的Bug
前言: 话说昨晚还是前晚,写了一篇:讲述Sagit.Framework解决:双向引用导致的IOS内存泄漏(上) 文章写到最后时,多了很多莫名奇妙的问题!!! 为了解决了这些莫名奇妙的问题,我又战斗了2 ...
- 讲述Sagit.Framework解决:双向引用导致的IOS内存泄漏(下)- block中任性用self
前言: 在处理完框架内存泄漏的问题后,见上篇:讲述Sagit.Framework解决:双向引用导致的IOS内存泄漏(中)- IOS不为人知的Bug 发现业务代码有一个地方的内存没释放,原因很也简单: ...
- iOS内存泄漏自动检测工具PLeakSniffer
新款objective-C内存泄漏自动检测工具 PLeakSniffer , GitHub地址 (https://github.com/music4kid/PLeakSniffer). 背景 前些天读 ...
- 使用Xcode和Instruments调试解决iOS内存泄漏
尽管iOS 5.0加入版本号之后ARC机制,由于相互引用关系是复杂的.内存泄漏可能仍然存在.于是,懂原理是非常重要的. 这里讲述在没有ARC的情况下,怎样使用Instruments来查找程序中的内存泄 ...
- 经验之道:最有效的iOS内存泄漏检测
版权声明:本文由胡涛原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/125 来源:腾云阁 https://www.qclou ...
- iOS 内存泄漏监测自动化
在 Android 上,Square 这家公司提供了非常有名的工具: leakcanary ,来帮助开发者们在日常开发过程中就能够发现内存泄漏.但在 iOS 上呢?在 Google 的时候,我发现了两 ...
- 一起来看看IOS内存泄漏的一个问题
很多iOS开发的朋友都是比较关心内存泄漏的问题,在实际的开发工作中首先我们需要知道程序有没有内存泄露,然后定位到底是哪行代码出现内存泄露了,这样才能将其修复.最简单的方法当然是借助于专业的检测工具,比 ...
- IOS内存泄漏
1. . - (void)viewDidLoad { [superviewDidLoad]; self.view.frame=CGRectMake(, , , ); NSArray *title1=[ ...
- iOS 内存泄漏排查以及处理
使用Xcode7的Instruments检测解决iOS内存泄露 文/笨笨的糯糯(简书作者)原文链接:http://www.jianshu.com/p/0837331875f0作为一名iOS开发攻城 ...
随机推荐
- 简易封装手机浏览器touch事件
做手机开发时候,简单想用一些动作,如touchLeft,touchRight等, 使用其他库文件就要加载很多不必要的东西,流量的浪费 今天简单写了封装touch的库,简单的监听一些逻辑 onTouch ...
- python之字典常用语法
1. 创建字典 描述:生成字典 语法: dic={'k1':'v1'} 样例: dic=dict(k1='v1',k2='v2') dic={'k1':'v1','k2':'v2'} 2. 取键值ge ...
- 矢量切片(Vector tile)番外一:Proj4js
说明:番外篇是对正篇矢量切片(Vector tile)中提到的一些值得继续延伸的关注点继续进行探索和学习,所涉及的内容以解决实际问题为主要导向. 一.新的需求? 在完成了矢量切片的工作后,新的需求出现 ...
- Ioc容器BeanPostProcessor-Spring 源码系列(3)
Ioc容器BeanPostProcessor-Spring 源码系列(3) 目录: Ioc容器beanDefinition-Spring 源码(1) Ioc容器依赖注入-Spring 源码(2) Io ...
- [Hadoop] - TaskTracker源码分析(状态发送)
TaskTracker节点向JobTracker汇报当前节点的运行时信息时候,是将运行状态信息同心跳报告一起发送给JobTracker的,主要包括TaskTracker的基本信息.节点资源使用信息.各 ...
- JavaScript中国象棋程序(5) - Alpha-Beta搜索
"JavaScript中国象棋程序" 这一系列教程将带你从头使用JavaScript编写一个中国象棋程序.这是教程的第5节. 这一系列共有9个部分: 0.JavaScript中国象 ...
- web请求
概述 发起一个http请求的过程就是建立一个socket通信的过程. 我们可以模仿浏览器发起http请求,譬如用httpclient工具包,curl命令等方式. curl "http://w ...
- gridView 编辑单元格获取单元格焦点位置(位于单元格的焦点位置)
1.主要代码: private void Form1_Load(object sender, EventArgs e) { DataTable dt = new DataTable(); dt.Col ...
- C# Windows 异步线程
Task t = new Task(new Action(() => { //推送产品 ...
- Linux SVN安装部署
系统:centos6.3 svn: subversion-1.6.1 apache: httpd-2.2.29 //创建svn路径 [root@localhost /]# mkdir svn [roo ...