美文网首页
iOS-网络请求(AFN)和缓存(YYCaChe)的二次封装

iOS-网络请求(AFN)和缓存(YYCaChe)的二次封装

作者: 长衣貌 | 来源:发表于2019-03-26 09:59 被阅读0次

    项目中常用的网络请求和数据缓存封装

    需要AFN和YYCaChe
    有自定义log和提示框,根据需要替换就可以了.

    1.数据缓存
    声明部分

    //
    //  NetworkCache.h
    //  BuLuoKe
    //
    //  Created by 裴铎 on 2018/9/4.
    //  Copyright © 2018年 裴铎. All rights reserved.
    //
    
    #import <Foundation/Foundation.h>
    
    @interface NetworkCache : NSObject
    
    /** 将数据缓存并设置key */
    + (void)setHTTPCache:(id)httpData URLString:(NSString *)urlString parameters:(NSDictionary *)parameters;
    
    /** 通过key获取缓存数据 */
    + (id)getHTTPCacheForURLString:(NSString *)urlString parameters:(NSDictionary *)parameters;
    
    /** 获取所有缓存数据的大小 */
    + (NSInteger)getAllCacheDataSize;
    
    /** 清除所有缓存 */
    + (void)clearAllCacheDatas;
    
    @end
    
    

    实现部分

    //
    //  NetworkCache.m
    //  BuLuoKe
    //
    //  Created by 裴铎 on 2018/9/4.
    //  Copyright © 2018年 裴铎. All rights reserved.
    //
    
    #import "NetworkCache.h"
    #import "YYCache.h" //数据缓存第三方类
    
    
    /** 用于缓存数据的第三方对象 */
    static YYCache * _defaultYYCache = nil;
    /** 用于缓存数据的文件名 */
    static NSString * const NetworkResponseCache = @"NetworkResponseCache";
    
    @implementation NetworkCache
    
    #pragma mark - 私有方法
    
    /** 类内存第一次加载时会调用下面两个方法(load, initialize),只加载一次 */
    //+ (void)load{
    //    
    //}
    /** 类的初始化方法 */
    + (void)initialize{
        /** 实力化YYCache对象 */
        _defaultYYCache = [YYCache cacheWithName:NetworkResponseCache];
    }
    
    /** 根据url和参数字典来拼接成一个缓存用的key */
    + (NSString *)cacheKeyWithURLString:(NSString *)urlString parameters:(NSDictionary *)parameters{
        if (parameters.count == 0 || parameters == nil) {
            return urlString;
        }
        /** 将参数字典转成字符窜 */
        //1./
        NSData * stringData = [NSJSONSerialization dataWithJSONObject:parameters options:0 error:nil];
        //2./
        NSString * parametersString = [[NSString alloc] initWithData:stringData encoding:NSUTF8StringEncoding];
        /** 将url和参数进行拼接, 并返回 */
        return [urlString stringByAppendingString:parametersString];
    }
    
    #pragma mark - 对外接口
    
    /** 将数据缓存并设置key */
    + (void)setHTTPCache:(id)httpData URLString:(NSString *)urlString parameters:(NSDictionary *)parameters {
        /** 拼接key */
        NSString * key = [self cacheKeyWithURLString:urlString parameters:parameters];
        /** 缓存数据 */
        [_defaultYYCache setObject:httpData forKey:key];
    }
    
    /** 通过key获取缓存数据 */
    + (id)getHTTPCacheForURLString:(NSString *)urlString parameters:(NSDictionary *)parameters{
        /** 拼接key */
        NSString * key = [self cacheKeyWithURLString:urlString parameters:parameters];
        /** 通过key获取缓存 */
        return [_defaultYYCache objectForKey:key];
    }
    
    /** 获取所有缓存数据的大小 */
    + (NSInteger)getAllCacheDataSize{
        /** 返回缓存的大小 */
        return [_defaultYYCache.diskCache totalCost];
    }
    
    /** 清除所有缓存 */
    + (void)clearAllCacheDatas{
        [_defaultYYCache.diskCache removeAllObjects];
    }
    
    @end
    
    

    2.网络请求
    声明文件

    //
    //  RequestDATA.h
    //  XiMaLaYa
    //
    //  Copyright © 2018年 com,baweijiaoyu. All rights reserved.
    // 网络数据请求类
    
    #import <Foundation/Foundation.h>
    
    typedef void(^UPLoadDataPass)(id resultData, NSError * error );
    
    /** 缓存策略枚举, 默认是忽略缓存, 直接请求网络数据 */
    typedef NS_ENUM(NSInteger, HTTPCacheType) {
        /** 忽略缓存, 直接请求网络数据 */
        ClientReloadIgnoringLocalCacheData = 0,
        /** 先读取缓存数据, 然后马上同步网络数据 */
        ClientReturnCacheDataThenLoad = 1,
        /** 有缓存数据时, 使用缓存,不再请求网络数据, 没有缓存才请求网络数据,(数据总也不变时用) */
        ClientReturnCacheDataElseLoad = 2,
        /** 有缓存就用缓存,没有也不发送请求,(直接当成请求失败处理) 用于断网情况 */
        ClientReturnCacheDataDontLoad = 3
    };
    
    /** 请求数据策略枚举 */
    typedef NS_ENUM(NSInteger, HTTPRequestType) {
        GET = 0,
        POST = 1
    };
    
    /** 用于数据回传的block */
    typedef void(^HTTPDataPass)(id resultData, NSError * error );
    
    @interface RequestDATA : NSObject
    +(instancetype)getInstance;
    /**
     网络数据的请求方式
     参数列表:
     1./请求数据的方式(GET、POST)
     2./请求的网址字符串
     3./请求的参数
     4./缓存策略
     5./请求结束回调
     */
    + (void)request:(HTTPRequestType)type
          urlString:(NSString *)urlString
         parameters:(NSDictionary *)parameters
        cachePolicy:(HTTPCacheType)cachePolicy
         completion:(__strong HTTPDataPass)completion;
    
    /** 获取所有缓存数据的大小 */
    + (NSInteger)getAllCacheDataSize;
    
    /** 清除所有缓存 */
    + (void)clearAllCacheDatas;
    
    @end
    

    实现文件

    //
    //  RequestDATA.m
    //  XiMaLaYa
    //
    //  Copyright © 2018年 com,baweijiaoyu. All rights reserved.
    //// 网络数据请求类 
    
    #import "RequestDATA.h"
    #import "defind.h" //全局宏
    #import "UIView+Alert.h"
    #import "AFNetworking.h"
    #import "NetworkCache.h" //对YYCache的二次封装
    
    @interface RequestDATA ()
    
    /** AFN的请求对象 */
    @property (nonatomic, strong) AFHTTPSessionManager * manager;
    /** 是否发送网络数据 */
    @property (nonatomic, assign, getter=isReloadData) BOOL reloadData;
    
    @end
    
    
    @implementation RequestDATA
    
    
    #pragma mark - 缓存处理
    
    /** 获取所有缓存数据的大小 */
    + (NSInteger)getAllCacheDataSize{
        return [NetworkCache getAllCacheDataSize];
    }
    
    /** 清除所有缓存 */
    + (void)clearAllCacheDatas{
        [NetworkCache clearAllCacheDatas];
    }
    
    /** 获取磁盘缓存数据 */
    - (void)getCacheDataWithURLString:(NSString *)urlString
                         parameters:(NSDictionary *)parameters
                        cachePolicy:(HTTPCacheType)cachePolicy
                         completion:(HTTPDataPass)completion{
        
        /** 获取缓存 */
        id cacheData = [NetworkCache getHTTPCacheForURLString:urlString parameters:parameters];
        /** 缓存策略区分 */
        if (cachePolicy == ClientReturnCacheDataThenLoad) {
            /** 是否允许请求数据 */
            self.reloadData = YES;
        }else if (cachePolicy == ClientReturnCacheDataElseLoad) {
            /** 判断是否有缓存 */
            if (cacheData == nil) {
                /** 是否允许请求数据 */
                self.reloadData = YES;
            }else{
                /** 是否允许请求数据 */
                self.reloadData = NO;
            }
        }else if (cachePolicy == ClientReturnCacheDataDontLoad) {
            /** 是否允许请求数据 */
            self.reloadData = NO;
        }
    
        /** 返回缓存数据 */
        completion(cacheData, nil);
    }
    
    #pragma mark - 对外的数据请求接口
    
    + (void)request:(HTTPRequestType)type
          urlString:(NSString *)urlString
         parameters:(NSMutableDictionary *)parameters
        cachePolicy:(HTTPCacheType)cachePolicy
         completion:(HTTPDataPass)completion{
    //    parameters[@"user_token"] = UserToken;
        /** 创建一个本类的对象, 用来调用本类的私有实力方法 */
        RequestDATA * request = [[RequestDATA alloc] init];
        /** 判断缓存策略 */
        if (cachePolicy != ClientReloadIgnoringLocalCacheData) {
            /** 获取磁盘缓存 */
            [request getCacheDataWithURLString:urlString parameters:parameters cachePolicy:cachePolicy completion:completion];
            /** 判断是否允许请求数据, 如果不允许就直接返回 */
            if (request.isReloadData == NO) {
                return ;
            }
        }
        /** 开始请求网络数据之前显示指示器 */
        dispatch_async(dispatch_get_main_queue(), ^{/** 回到主线程 */
            [K_Key_Window showIndicatorView];
        });
    
        /** 判断请求的方式 */
        if (type == 0) {
            [request getDataWithURLString:urlString parameters:parameters cachePolicy:cachePolicy complement:^(id resultData, NSError *error) {
                /** 返回结果 */
                if (error != nil) {
                    K_Log(@"\n================错误信息===========================\n%@", error)
                    dispatch_async(dispatch_get_main_queue(), ^{/** 回到主线程 */
                        [K_Key_Window showMBHUDAlertWithMessage:@"服务器发生错误了!~" hideTimeInterval:1.0];
                    });
                }
                completion(resultData, error);
                /** 请求数据结束时,停止指示器 */
                dispatch_async(dispatch_get_main_queue(), ^{/** 回到主线程 */
                    [K_Key_Window hideIndicatorView];
                });
            }];
        }else if (type == 1) {
            [request postDataWithURLString:urlString parameters:parameters cachePolicy:cachePolicy complement:^(id resultData, NSError *error) {
                /** 返回结果 */
                if (error != nil) {
                    K_Log(@"\n================错误信息===========================\n%@", error)
                    dispatch_async(dispatch_get_main_queue(), ^{/** 回到主线程 */
                        [K_Key_Window showMBHUDAlertWithMessage:@"服务器发生错误了!~" hideTimeInterval:1.0];
                    });
                }
                completion(resultData, error);
                /** 请求数据结束时,停止指示器 */
                dispatch_async(dispatch_get_main_queue(), ^{/** 回到主线程 */
                    [K_Key_Window hideIndicatorView];
                });
            }];
        }
    }
    
    #pragma mark - 私有的实际请求方法
    
    - (void)getDataWithURLString:(NSString *)urlString parameters:(NSDictionary *)parameters cachePolicy:(HTTPCacheType)cachePolicy complement:(HTTPDataPass)block{
        /** 对特殊字符转吗 */
    //    NSString * urlStr = [urlString  stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet characterSetWithCharactersInString:@"`#%^{}\"[]|\\<> "].invertedSet];
    //    NSString * urlStr = [urlString stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
        //NSString * urlStr = [urlString stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]];
        NSString * urlStr = [urlString stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLQueryAllowedCharacterSet]];
        /** 发送网络请求 */
        [self.manager GET:urlStr parameters:parameters progress:^(NSProgress * _Nonnull downloadProgress) {
            /** 进度回调 */
        } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
            /** 成功回调 */
            /** 缓存数据, 做容错处理 */
            if (responseObject != nil) {
                /** 返回数据给外界 */
                block(responseObject, nil);
                /** 判断是否需要缓存 */
                if (cachePolicy != ClientReloadIgnoringLocalCacheData) {
                    [NetworkCache setHTTPCache:responseObject URLString:urlString parameters:parameters];
                }
            }
            
        } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
            /** 失败回调 */
            block(nil, error);
        }];
    }
    +(instancetype)getInstance {
        RequestDATA *net = [[RequestDATA alloc] init];
        return net;
    }
    - (void)postDataWithURLString:(NSString *)urlString parameters:(NSDictionary *)parameters cachePolicy:(HTTPCacheType)cachePolicy complement:(HTTPDataPass)block{
        /** 发送网络请求 */
        [self.manager POST:urlString parameters:parameters progress:^(NSProgress * _Nonnull uploadProgress) {
            /** 进度回调 */
        } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
            /** 成功回调 */
            /** 缓存数据, 做容错处理 */
            if (responseObject != nil) {
                /** 返回数据给外界 */
                block(responseObject, nil);
                /** 判断是否需要缓存 */
                if (cachePolicy != ClientReloadIgnoringLocalCacheData) {
                    [NetworkCache setHTTPCache:responseObject URLString:urlString parameters:parameters];
                }
            }
        } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
            /** 失败回调 */
            block(nil, error);
        }];
    }
    
    /** 懒加载, 并设置请求的支持格式 */
    - (AFHTTPSessionManager *)manager{
        if (_manager == nil) {
            /** 请求对象 */
            _manager =[AFHTTPSessionManager manager];
            /** 设置响应
             响应解析器, 这样返回的数据就是解析好的JSON数据, 不需要再做反序列化 */
            _manager.responseSerializer = [AFJSONResponseSerializer serializerWithReadingOptions:NSJSONReadingAllowFragments];
            /** 支持的数据格式 */
            NSSet * set = [NSSet setWithObjects:
                           @"application/json",
                           @"text/json",
                           @"text/javascript",
                           @"text/html",
                           @"text/plain",
                           @"image/jpg",
                           nil];
            _manager.responseSerializer.acceptableContentTypes = set;
            /** 设置请求 */
            _manager.requestSerializer = [AFHTTPRequestSerializer serializer];
            /** 将用户的token放在请求头中发送给后台 */
    //        [_manager.requestSerializer setValue:K_Read_UserToken forHTTPHeaderField:@"token"];
            /** 设置请求超时时间, AFN默认是60秒 */
            [_manager.requestSerializer willChangeValueForKey:@"timeoutInterval"];
            _manager.requestSerializer.timeoutInterval = 15.0f;
            [_manager.requestSerializer didChangeValueForKey:@"timeoutInterval"];
            /** AF自动处理返回NULL对象的异常 */
            AFJSONResponseSerializer * jsonResponse = (AFJSONResponseSerializer *)_manager.responseSerializer;
            jsonResponse.removesKeysWithNullValues = YES;
        }
        return _manager;
    }
    
    @end
    
    

    相关文章

      网友评论

          本文标题:iOS-网络请求(AFN)和缓存(YYCaChe)的二次封装

          本文链接:https://www.haomeiwen.com/subject/nobwvqtx.html