iOS AFHTTPSessionManager内存泄漏解决方案-2021.11

起因:
AFURLSessionManager实现了NSURLSession的协议,即AFURLSessionManager和NSURLSession互相持有,如果这个delegate是week的话,那没什么问题,但是系统提供的是retain:如图


image.png

所以里面是有循环引用的。一旦使用非单例的方式来使用AFHTTPSessionManager,如果不做特殊处理,就会导致这个对象在应用存活期间是不会销毁的,如果应用网络请求过多,用户反复进入各种vc,内存泄漏明显。这明显不安全。

解决方案一:把该manager封装成单例
方案代码:

static AFHTTPSessionManager *manager;

/* 封装成 单例会话管理者 */
+ (AFHTTPSessionManager *)sharedManager {
    
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        // 初始化请求管理类
        manager = [AFHTTPSessionManager manager];
        manager.requestSerializer = [AFJSONRequestSerializer serializer];
        
        // 设置15秒超时 - 取消请求
        manager.requestSerializer.timeoutInterval = 15.0;
        // 编码
        manager.requestSerializer.stringEncoding = NSUTF8StringEncoding;
        // 缓存策略
        manager.requestSerializer.cachePolicy = NSURLRequestReloadIgnoringLocalCacheData;
        
        manager.responseSerializer = [AFJSONResponseSerializer serializer];
        // 支持内容格式
        manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json", @"text/plain", @"text/javascript", @"text/json", @"text/html", nil];
    });
    return manager;
}

问题:很明显,同一时刻只能有一个网络请求。异步会有问题。当两个线程同时申请manager对象时,肯定有一个manager申请不到,无法网络请求

解决方案二:在网络请求的block内把task取消掉
无论是success,还是failure的回调都取消掉,当然在block外部需要弱化一下manager对象

__weak typeof(manager) weakManager = manager;

然后在两个回调方法里加上

[weakManager invalidateSessionCancelingTasks:YES resetSession:YES];

具体代码如下:


image.png

两种方案都可以解决内存泄漏问题。推荐第二种,封装在某个网络请求基类里,或者每次网络请求结束都写上。

你可能感兴趣的:(iOS AFHTTPSessionManager内存泄漏解决方案-2021.11)