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开发攻城 ...
随机推荐
- java二维数组学习(转)
转自:http://blog.csdn.net/java1992/article/details/5808709,在这里谢过了 /* * java学习: * 二维数组的使用: */public cla ...
- ZeroMQ初探
概述 ZeroMQ(也称为 ØMQ,0MQ 或 zmq)是一个可嵌入的网络通讯库(对 Socket 进行了封装). 它提供了携带跨越多种传输协议(如:进程内,进程间,TCP 和多播)的原子消息的 so ...
- Unity预计算光照的学习(速度优化,LightProb,LPPV)
1.前言 写这篇文章一方面是因为unity的微博最近出了关于预计算光照相关的翻译文章,另一方面一些美术朋友一直在抱怨烘培速度慢 所以抱着好奇的心态来学习一下unity5的PRGI预计算实时光照 2.基 ...
- MVC中登录页图片验证码总结
直接上代码了 using System;using System.Collections.Generic;using System.Drawing;using System.Drawing.Imagi ...
- window7 32位安装Oracle11g
http://www.cnblogs.com/wangsaiming/p/3573509.html
- c++ 调用dl里的导出类
来源:http://blog.csdn.net/yysdsyl/article/details/2626033 动态dll的类导出:CPPDll2->test.h #pragma once // ...
- Winfrom 程序打包及安装
前言 近期被公司外派到驻空调厂的项目组,支援一个TCP相关的程序对接.主要是做智能门禁系统,然后主要是统计出实时的进出人数. 我这边能作为服务端,门禁设备作为客户端,整个流程并不算复杂,根据协议来写, ...
- windows 2003装.net 4.0时提示 WIC windows Imaging Component
运行此安装程序之前,必须安装32位windows映像处理组件(WIC) WIC windows Imaging Component下载地址: http://download.microsoft.com ...
- 深入理解ajax系列第四篇——请求实例
前面的话 在使用ajax的过程中,常用的请求方式是GET和POST两种.本文将以实例的形式来详细说明这两种请求方式 GET GET是最常见的请求类型,最常用于向服务器查询某些信息.必要时,可以将查询字 ...
- RabbitMQ(从安装到使用)
RabbitMQ 一,RabbitMQ简单介绍: RabbitMQ是一个在AMQP基础上完整的,可复用的企业消息系统.他遵循Mozilla Public License开源协议. MQ全称为Messa ...