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
posted @ 2017-03-30 13:45  豆丶浆油条  阅读(1209)  评论(0编辑  收藏  举报