本篇博客是接着上一篇AFNetworking源码解析的后续,如果想对AFNetworking源码有所了解。

请读一下https://www.cnblogs.com/guohai-stronger/p/9191497.html

YTKNetwork的源码详解:https://www.cnblogs.com/guohai-stronger/p/9194519.html

这篇博客主要是对AFNetworking代码进行封装,也是自己在实际项目中所使用过的(对中小型企业或者项目)可直接使用,也会在github上传(偷偷窃喜一下,一直想在github上有所项目,一直没有时间),趁着端午几天有时间,给github贡献点自己力量。本篇读下来大约10-15分钟,欢迎指正。

一.结构解析

最核心(可直接拖拽到新项目中)

(拓展cocoapods的拖入) 拖入之前,需要导入AFNetworking,最好利用cocoapods导入,cocoapods导入时

vim Podfile     --->输入i
platform :ios, '8.0'

target "targetName" do

pod 'AFNetworking'

end

(有一点在然后按Esc,并且输入“ :”号进入vim命令模式,然后在冒号后边输入wq)最后pod install。编译一下,拖入项目的类就好了,就不会编译错误。

二.分类讲解

1.Singleton

#import <UIKit/UIKit.h>
////////////////////////////////////////////////////////////////////////////////
// 申明和定义单例对象:
// AS_SINGLETON( XxxClass );
// DEF_SINGLETON( XxxClass );
#undef AS_SINGLETON
#define AS_SINGLETON( __class ) \
- (__class *)sharedInstance; \
+ (__class *)sharedInstance; #undef DEF_SINGLETON
#define DEF_SINGLETON( __class ) \
- (__class *)sharedInstance \
{ \
return [__class sharedInstance]; \
} \
+ (__class *)sharedInstance \
{ \
static dispatch_once_t once; \
static __class * __singleton__; \
dispatch_once( &once, ^{ __singleton__ = [[[self class] alloc] init]; } ); \
return __singleton__; \
}

通过Singleton,来声明和定义单例对象。

声明单例

// 声明单例
AS_SINGLETON(APIClient);

定义单例

// 定义单例
DEF_SINGLETON(APIClient);

2.APIClient.h

(1)请求响应的状态

// 请求响应的状态
typedef NS_ENUM(NSInteger, ApiRequestStatusCode) {
ApiRequestNotReachable = -, // 当前网络已断开
ApiRequestErr = -, // 请求出错
ApiRequestOK = , // 请求成功
};

当时的请求响应状态只有三个,故定义一个枚举表示请求响应的状态。

(2)请求的方法类型

// 请求的方法类型
typedef NS_ENUM(NSInteger, HttpMethod) {
GET = ,
POST = ,
DELETE = ,
PUT = ,
PATH_GET = , // GET请求,填充参数到url上,处理@"user/account/check/{phone}",{phone}这种情况
QUERY_GET = , // GET请求,填充query参数到url上,处理"?a=1&b=2"这种情况
};

(3)请求相应的block

// 请求响应Block
typedef void (^APIClientRequestResponse)(ApiRequestStatusCode requestStatusCode, id JSON);

(4)通过设置代理拦截成功响应数据

@protocol APIClientDelegate <NSObject>

@required
// 拦截成功请求响应数据进行处理
- (void)handleSuccessRequest:(id)JSON completion:(void(^)(id aJSON))cb; @end

在.h中还有一些方法属性

// 为处理拦截的响应数据而设置的委托
@property (nonatomic, weak) id<APIClientDelegate> delegate; // 声明单例
AS_SINGLETON(APIClient); // 获取当前的网络状态类型(返回:0-No wifi or cellular(无网络), 1-2G, 2-3G, 3-4G, 4-LTE, 5-wifi)
+ (int)networkType; // 网络状态监听,应用当前是否有网络:有返回YES,没有返回NO
+ (void)networkReachableWithBlock:(void(^)(BOOL isReachable))block; // 发送请求,返回JSON格式的响应数据
+ (void)requestURL:(NSString *)urlString
httpMethod:(HttpMethod)method
params:(NSDictionary *)params
response:(APIClientRequestResponse)response; // 取消掉所有网络请求
+ (void)cancelAllRequest;

3.APIClient.m

(1)创建请求管理对象

@property (nonatomic, strong) AFHTTPSessionManager *manager;

(2)获取当前的网络状态类型

// 获取当前的网络状态类型(返回:0-No wifi or cellular(无网络), 1-2G, 2-3G, 3-4G, 4-LTE, 5-wifi)
+ (int)networkType {
NSArray *subviews = [[[[UIApplication sharedApplication] valueForKey:@"statusBar"] valueForKey:@"foregroundView"]subviews];
NSNumber *dataNetworkItemView = nil; for (id subview in subviews) {
if([subview isKindOfClass:[NSClassFromString(@"UIStatusBarDataNetworkItemView") class]]) {
dataNetworkItemView = subview;
break;
}
}
int ret = ;
switch ([[dataNetworkItemView valueForKey:@"dataNetworkType"] integerValue]) {
case :
ret = ;
break;
case :
ret = ;
break;
case :
ret = ;
break;
case :
ret = ;
break;
case :
ret = ;
break;
case :
ret = ;
break;
default:
break;
}
return ret;
}

(3)网络状态监听,应用当前是否有网络

// 网络状态监听,应用当前是否有网络
+ (void)networkReachableWithBlock:(void(^)(BOOL isReachable))block {
[[AFNetworkReachabilityManager sharedManager] startMonitoring]; [[AFNetworkReachabilityManager sharedManager] setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {
switch (status) {
case AFNetworkReachabilityStatusReachableViaWWAN:
case AFNetworkReachabilityStatusReachableViaWiFi: {
if (block) {
block(YES);
}
break;
}
case AFNetworkReachabilityStatusNotReachable: {
if (block) {
block(NO);
}
break;
}
default:
break;
}
}]; //结束监听
//[[AFNetworkReachabilityManager sharedManager] stopMonitoring];
}

(3)请求最重要最核心的代码

// 发送请求,返回JSON格式的响应数据
+ (void)requestURL:(NSString *)urlString
httpMethod:(HttpMethod)method
params:(NSDictionary *)params
response:(APIClientRequestResponse)response {
if ([APIClient networkType] > ) {
APIClient *client = [APIClient sharedInstance];
if (!client.manager) {
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
manager.requestSerializer.timeoutInterval = ;
manager.requestSerializer = [AFHTTPRequestSerializer serializer];
manager.responseSerializer = [AFJSONResponseSerializer serializer];
AFJSONResponseSerializer *responseSerializer = [AFJSONResponseSerializer serializerWithReadingOptions:NSJSONReadingAllowFragments];
manager.responseSerializer = responseSerializer;
manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json", @"text/json", @"text/javascript",@"text/html",@"text/plain",@"application/x-www-form-urlencodem", nil]; client.manager = manager;
} __weak typeof(self)weakSelf = self;
switch (method) {
case PATH_GET: {
urlString = [APIClient pathGet:urlString params:params];
// //LOG(@"PATH_GET http_url:%@",urlString);
[client.manager GET:urlString
parameters:nil
progress:nil
success:^(NSURLSessionDataTask * __unused task, id JSON) {
__strong typeof(weakSelf)strongSelf = weakSelf;
if (strongSelf) {
[strongSelf handleSuccessRequest:JSON cb:response];
}
}
failure:^(NSURLSessionDataTask *__unused task, NSError *error) {
if (response) {
response(ApiRequestErr, nil);
}
}];
break;
}
case QUERY_GET: {
urlString = [APIClient queryGet:urlString params:params];
//LOG(@"QUERY_GET http_url:%@",urlString);
[client.manager GET:urlString
parameters:nil
progress:nil
success:^(NSURLSessionDataTask * __unused task, id JSON) {
__strong typeof(weakSelf)strongSelf = weakSelf;
if (strongSelf) {
[strongSelf handleSuccessRequest:JSON cb:response];
}
}
failure:^(NSURLSessionDataTask *__unused task, NSError *error) {
if (response) {
response(ApiRequestErr, nil);
}
}];
break;
}
case GET: {
// //LOG(@"GET http_url:%@",urlString);
[client.manager GET:urlString
parameters:nil
progress:nil
success:^(NSURLSessionDataTask * __unused task, id JSON) {
__strong typeof(weakSelf)strongSelf = weakSelf;
if (strongSelf) {
[strongSelf handleSuccessRequest:JSON cb:response];
}
}
failure:^(NSURLSessionDataTask *__unused task, NSError *error) {
if (response) {
response(ApiRequestErr, nil);
//LOG(@"GET http_url:%@",urlString);
}
}];
break;
}
case POST: {
// //LOG(@"POST http_url:%@",urlString);
[client.manager POST:urlString
parameters:params
progress:nil
success:^(NSURLSessionDataTask * __unused task, id JSON) {
__strong typeof(weakSelf)strongSelf = weakSelf;
if (strongSelf) {
[strongSelf handleSuccessRequest:JSON cb:response];
}
}
failure:^(NSURLSessionDataTask *__unused task, NSError *error) {
// //LOG(@"%@",error);
if (response) {
response(ApiRequestErr, nil);
}
}];
break;
}
case DELETE: {
//LOG(@"DELETE http_url:%@",urlString);
[client.manager DELETE:urlString
parameters:nil
success:^(NSURLSessionDataTask * __unused task, id JSON) {
__strong typeof(weakSelf)strongSelf = weakSelf;
if (strongSelf) {
[strongSelf handleSuccessRequest:JSON cb:response];
}
}
failure:^(NSURLSessionDataTask *__unused task, NSError *error) {
if (response) {
response(ApiRequestErr, nil);
}
}];
break;
}
case PUT: {
//LOG(@"PUT http_url:%@",urlString);
[client.manager PUT:urlString
parameters:params
success:^(NSURLSessionDataTask * __unused task, id JSON) {
__strong typeof(weakSelf)strongSelf = weakSelf;
if (strongSelf) {
[strongSelf handleSuccessRequest:JSON cb:response];
}
}
failure:^(NSURLSessionDataTask *__unused task, NSError *error) {
if (response) {
response(ApiRequestErr, nil);
}
}];
break;
}
}
} else { // 没有连接网络
if (response)
response(ApiRequestNotReachable, nil);
}
}

(4)统一处理响应数据

// 统一处理响应数据
+ (void)handleSuccessRequest:(id)JSON cb:(void(^)(ApiRequestStatusCode requestStatusCode, id JSON))cb {
APIClient *client = [APIClient sharedInstance];
if (client.delegate && [client.delegate respondsToSelector:@selector(handleSuccessRequest:completion:)]) {
[client.delegate handleSuccessRequest:JSON
completion:^(id aJSON) {
if (cb) {
cb(ApiRequestOK, aJSON);
}
}];
} else {
if (cb) {
cb(ApiRequestOK, JSON);
}
}
}

(5)参数拼接

// 填充参数到url上,处理@"user/account/check/{phone}",{phone}这种情况
+ (NSString *)pathGet:(NSString *)uri
params:(NSDictionary *)params {
if (nil == uri|| nil == params || params.count <= ) {
return uri;
}
for (NSString *key in params) {
NSString *key2 = [NSString stringWithFormat:@"{%@}",key];
if ([uri rangeOfString:key2].location != NSNotFound) {
uri = [uri stringByReplacingOccurrencesOfString:key2 withString:params[key]];
}
}
return [uri stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
} // 填充query参数到url上,处理"?a=1&b=2"这种情况
+ (NSString *)queryGet:(NSString *)uri
params:(NSDictionary *)params {
if (nil == uri || nil == params || params.count <= ) {
return uri;
}
NSMutableString *tmpUri = [NSMutableString stringWithString:uri];
int i = ;
for (NSString *key in params) {
if (i == ) {
[tmpUri appendFormat:@"?%@=%@",key, params[key]];
} else {
[tmpUri appendFormat:@"&%@=%@",key, params[key]];
}
i++;
}
return [tmpUri stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];;
}

(6)取消所有网络请求

// 取消掉所有网络请求
+ (void)cancelAllRequest {
APIClient *client = [APIClient sharedInstance];
if (client.manager) {
if (client.manager.operationQueue) {
[client.manager.operationQueue cancelAllOperations];
}
}
}

上面就是代码的主要封装,可以应对中小型企业和App的要求。

三.使用

1.新建API.h继承NSObject

2.在API.m使用

3.在控制器里面调用如下

 WS(weakSelf);
if ([APIClient networkType] > ){
//获取个人界面的信息
[API1 getEvaluateMeDataWithCB:^(ApiRequestStatusCode requestStatusCode, id JSON) {
switch (requestStatusCode) {
case ApiRequestOK: {
//正确解析json,以及UI处理break;
}
}
case ApiRequestErr:
[MBProgressHUD showError:kErrTip];
break;
case ApiRequestNotReachable:
[MBProgressHUD showError:kNotReachableTip];
break;
default:
break;
}
}];
}else{
[MBProgressHUD showError:kNotReachableTip];
}

上面就是针对AFNetworking的简单封装,可以满足网络请求的封装。

本次代码的demo已上传git。github地址:https://github.com/zxy1829760/testAF

欢迎指正!!!

下一篇讲述另一篇关于网络封装的YTKNetwork源码解析。

AFNetworking封装-项目使用的更多相关文章

  1. iOS | AFNetworking封装

    为大家分享一个IOS处理网络请求,网络上传,网络下载等功能全面的一个第三方框架-AFNetworking,这是一个使用非常方便的网络框架. 最新的版本是基于NSURLSession,原来的NSURLC ...

  2. Android--带你一点点封装项目 MVP+BaseActivity+Retrofit+Dagger+RxJava(一)

    1,其实早就想把这些东西给封装封装的,一直没有时间,今天刚好项目进入到测试阶段了,Bug同事在哪儿测试的飞起,但发现提bug的尽然是我(得意脸),然后上午把ios的包测试了一下,顺便把服务器给测挂了( ...

  3. AFNetworking封装思路简析

    http://blog.csdn.net/qq_34101611/article/details/51698473 一.AFNetworking的发展 1. AFN 1.0版本 AFN 的基础部分是 ...

  4. C# .NET更智能的数据库操作封装项目

    前面两篇文章介绍了框架的思路及里面大概的实现过程,那时候忘记上传项目,就补发一下.顺便介绍下框架使用方式,并分析下框架使用的优缺点. 先发一下前两章的链接 篇一:http://www.cnblogs. ...

  5. Android--带你一点点封装项目 MVP+BaseActivity+Retrofit+Dagger+RxJava(三)

    1,这一篇博客是和大家一起来封装我们最后的Dagger2,其实之前也写过关于简单的Dagger2,这里是地址,完全没了解的同学可以先去看一下这篇,感谢很多小伙伴一直在耐心的等待这一篇 2,Dagger ...

  6. 基于AFNetworking封装的网络请求工具类【原创】

    今天给大家共享一个我自己封装的网络请求类,希望能帮助到大家. 前提,导入AFNetworking框架, 关于修改AFN源码:通常序列化时做对text/plan等的支持时,可以一劳永逸的修改源代码,在a ...

  7. Android--带你一点点封装项目 MVP+BaseActivity+Retrofit+Dagger+RxJava(二)

    1,昨天我们基本上把MVP给封装起来了,今天接着昨天的东西来结合RxJava把Retrofit把网络框架简单的封装一下,先看一下我们今天实现的效果: 哈哈 ,还是昨天的效果,好吧 ,我认错. 2,由于 ...

  8. 基于AFNetWorking封装一个网络请求数据的类

    1.新建一个继承于NSObject类的类,在.h文件中 #import "AFHTTPRequestOperationManager.h" //定义两个block来接收请求成功和失 ...

  9. 单例:用AFNetworking封装举例

    创建单例的方式 import AFNetworking class NetworkTools: AFHTTPSessionManager { //let是线程安全的 //使用单例实例只需要拿到Netw ...

随机推荐

  1. php实现最简单的MVC框架实例教程

    本文以一个实例的形式讲述了PHP实现MVC框架的过程,比较浅显易懂.现分享给大家供大家参考之用.具体分析如下: 首先,在学习一个框架之前,基本上我们都需要知道什么是mvc,即model-view-co ...

  2. android-audioRecord

    android 录音功能 录音的大致流程,流程图可以在文件下载:mediarecord.vsdx 切换设备.谁去更新播放流,自动选择新的设备?流程?

  3. File(File f, String child) File(String parent, String child)

    (转载)File(File f, String child) 根据f 抽象路径名和 child 路径名字符串创建一个新 File 实例. f抽象路径名用于表示目录,child 路径名字符串用于表示目录 ...

  4. php中出现乱码

    对于初学着来说,编辑中文php时,会出现乱码 在php代码中加入 随后在浏览器中,就会看到如下页面 这样就解决了php 中文乱码的问题.

  5. 【ProtoBuffer】windows上安装ProtoBuffer3.1.0 (附已编译资源)

    ------- 17.9.17更新  --- 以下这些方法都是扯淡,对我的机器不适用,我后来花了最后成功安装并亲测可用的方法不是靠vs编过的,vs生成的库引入后函数全部报undefine refere ...

  6. JAVA RSA非对称加密详解[转载]

    一.概述1.RSA是基于大数因子分解难题.目前各种主流计算机语言都支持RSA算法的实现2.java6支持RSA算法3.RSA算法可以用于数据加密和数字签名4.RSA算法相对于DES/AES等对称加密算 ...

  7. 【SP26073】DIVCNT1 - Counting Divisors 题解

    题目描述 定义 \(d(n)\) 为 \(n\) 的正因数的个数,比如 \(d(2) = 2, d(6) = 4\). 令 $ S_1(n) = \sum_{i=1}^n d(i) $ 给定 \(n\ ...

  8. C++内存读写例子

    /************************************************************* // 声明头文件 #ifndef LyShark__h #define L ...

  9. 前端页面使用emoji

    概述 最近在工作中遇到了移动端emoji的显示问题,于是查找了一些相关资料,试验了一番,把心得记录下来,供以后开发时参考,相信对其他人也有用. 参考资料: 为网站添加emoji表情的支持 思路 在页面 ...

  10. Mybatis框架七:三种方式整合Spring

    需要的jar包: 新建lib文件夹放入jar包,Build Path-->Add To Build Path之后: 原始Dao开发示例: src下:新建核心配置文件sqlMapConfig.xm ...