网络

Dream.png

https://github.com/starainDou 欢迎点星

AFNetWorking

  • 本段代码为总结知识点
// AFHTTPSessionManager *manager = [AFHTTPSessionManager manager]; //推荐下面方式
AFHTTPSessionManager *manager =[[AFHTTPSessionManager alloc] initWithBaseURL:BaseURL];
// 设置请求头
[manager.requestSerializer setValue:@"en" forHTTPHeaderField:@"locale"];
// 请求序列化
manager.requestSerializer = [AFJSONRequestSerializer serializer];
// 设置超时时间
manager.requestSerializer.timeoutInterval = 30.f;
// 响应内容序列化
manager.responseSerializer = [AFJSONResponseSerializer serializer];
// 支持的内容类型
manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects: @"application/json", @"text/json", @"text/plain",@"text/html", nil];
// 固定参数可这样写
NSDictionary *params = @{@"uid":userId,  @"token":token};
// 可选参数可这样写
NSMutableDictionary *params = [NSMutableDictionary dictionary];
if (userId) params[@"uid"] = userId;
if (token)  params[@"token"] = token;
// 网络请求
[manager POST:url parameters:params constructingBodyWithBlock:^(id formData) {
            [formData appendPartWithFileData:UIImageJPEGRepresentation(image, 0.5)
                                        name:@"uploadfile"
                                    fileName: fileName
                                    mimeType:@"image/jpeg"];
        } progress:^(NSProgress *uploadProgress) {
            
        } success:^(NSURLSessionDataTask *task, id responseObject) {
            NSLog(@"success : %@", responseObject);
        } failure:^(NSURLSessionDataTask *task, NSError *error) {
            NSLog(@"error : %@",error.localizedDescription);
        }];

NSURLSession

  • NSURLSession 的优势

    • NSURLSession 支持 http2.0 协议
    • 在处理下载任务的时候可以直接把数据下载到磁盘
    • 支持后台上传/下载
    • 同一个 session 发送多个请求,只需要建立一次连接(复用了TCP)
    • 提供了全局的 session 并且可以统一配置,使用更加方便
    • 下载的时候是多线程异步处理,效率更高
  • NSURLSessionTask 是一个抽象类,常用两个子类

    • NSURLSessionDataTask: 处理一般的网络请求,其子类NSURLSessionUploadTask处理上传请求
    • NSURLSessionDownloadTask,主要用于处理下载请求,有很大的优势
  • GET请求
    1.使用 NSURLSession 对象创建 Task
    2.执行 Task

    //创建 NSURLSession 对象
    NSURLSession *session = [NSURLSession sharedSession];
    /**
     根据对象创建 Task 请求
     url  方法内部会自动将 URL 包装成一个请求对象(默认是 GET 请求)
     completionHandler  完成之后的回调(成功或失败)
     param data     返回的数据(响应体)
     param response 响应头
     param error    错误信息
     */
    NSURLSessionDataTask *dataTask = [session dataTaskWithURL:url completionHandler:
                                      ^(NSData *data, NSURLResponse *response, NSError *error) {
                                          //解析服务器返回的数据
                                          NSLog(@"%@", [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]);
                                          //默认在子线程中解析数据
                                          NSLog(@"%@", [NSThread currentThread]);
                                      }];
    //发送请求(执行Task)
    [dataTask resume];
  • POST请求
    NSString *str = @"http://doudianyu.com/commetlist/";
    // 如果需要中文编码
    NSString *urlPath = [str stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLQueryAllowedCharacterSet]];
    //确定请求路径
    NSURL *url = [NSURL URLWithString:urlPath];
    //创建可变请求对象
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    //修改请求方法
    request.HTTPMethod = @"POST";
    // 设置请求头
    [request setValue:@"en" forHTTPHeaderField:@"local"];
    //设置请求体
    request.HTTPBody = [@"page=0&size=50&type=JSON" dataUsingEncoding:NSUTF8StringEncoding];
    //创建会话对象
    NSURLSession *session = [NSURLSession sharedSession];
    // 遮罩
    [SVProgressHUD show];
    //创建请求 Task
    NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request completionHandler:
                                      ^(NSData *data, NSURLResponse *response, NSError *error) {
                                          //回到主线程更新UI -> 撤销遮罩
                                          dispatch_async(dispatch_get_main_queue(), ^{
                                              [SVProgressHUD dismiss];
                                          });
                                          
                                          if (error) {
                                              NSLog(@"提示用户请求失败...error:%@",error);
                                          } else {
                                              // JSON解析 苹果原生效率很高
                                              NSDictionary *result = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableLeaves error:nil];
                                              if ([[result objectForKey:@"message"] isEqualToString:@"success"]) {
                                                  //获取数据->主线程更新UI
                                                  dispatch_async(dispatch_get_main_queue(), ^{
                                                      NSDictionary *data = [result objectForKey:@"data"];
                                                  });
                                              }else{
                                                  NSLog(@"未查到信息....");
                                              }
                                              NSLog(@"请求成功... %@",result);
                                          }
                                      }];
    //发送请求
    [dataTask resume];
  • NSURLSessionDataTask 设置代理发送请求
    1.创建 NSURLSession 对象设置代理
    2.使用 NSURLSession 对象创建 Task
    3.执行 Task
//确定请求路径
 NSURL *url = [NSURL URLWithString:@"http://doudianyu.com/commetlist/"];
 //创建可变请求对象
 NSMutableURLRequest *requestM = [NSMutableURLRequest requestWithURL:url];
 //设置请求方法
 requestM.HTTPMethod = @"POST";
 //设置请求体
 requestM.HTTPBody = [@"page=0&size=50&type=JSON" dataUsingEncoding:NSUTF8StringEncoding];
 //创建会话对象,设置代理
 /**
  第一个参数:配置信息
  第二个参数:设置代理
  第三个参数:队列,如果该参数传递nil 那么默认在子线程中执行
  */
 NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
                              delegate:self delegateQueue:nil];
 //创建请求 Task
 NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:requestM];
 //发送请求
 [dataTask resume];

4.遵守协议,实现代理方法(常用的有三种代理方法)

#pragma mark 接收到服务器响应的时候调用
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
     NSLog(@"子线程中执行 -- %@", [NSThread currentThread]);
     self.dataM = [NSMutableData data];
     // 默认情况下不接收数据,必须告诉系统是否接收服务器返回的数据
     completionHandler(NSURLSessionResponseAllow);
}
#pragma mark 接受到服务器返回数据的时候调用,可能被调用多次
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
     //拼接服务器返回的数据
     [self.dataM appendData:data];
}
#pragma mark 请求完成或者是失败的时候调用
-(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
     //解析服务器返回数据
     NSLog(@"%@", [[NSString alloc] initWithData:self.dataM encoding:NSUTF8StringEncoding]);
}

5.设置代理之后的强引用问题

  • NSURLSession对象使用时若设置了代理,session会对代理对象保持一个强引用,合适的时候应主动释放
  • dealloc中调用 invalidateAndCancel或finishTasksAndInvalidate方法释放对代理对象的强引用
    1.invalidateAndCancel 方法直接取消请求然后释放代理对象
    2.finishTasksAndInvalidate 方法等请求完成之后释放代理对象
    [self.session finishTasksAndInvalidate];
-(void)dealloc
{
    //注意:在不用的时候一定要调用该方法来释放,不然会出现内存泄露问题
    //方法一:取消所有过去的会话和任务
    [self.session invalidateAndCancel];
    //方法二:可在释放时做一些操作
    [self.session resetWithCompletionHandler:^{
         // 释放时做的操作
    }];
}
  • NSURLSessionDataTask 简单下载
[[[NSURLSession sharedSession] dataTaskWithURL:imgURL completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
     dispatch_async(dispatch_get_main_queue(), ^{
         self.imageView.image =  [UIImage imageWithData:data];
     });
 }] resume];
  • NSURLSessionDownloadTask 简单下载
    1.使用 NSURLSession 对象创建下载请求
    2.在下载请求中移动文件到指定位置
    3.执行请求
    4.代理方法监听下载进度
 //优点:该方法内部已经完成了边接收数据边写沙盒的操作,解决了内存飙升的问题
 NSURLSessionDownloadTask *downTask = [session downloadTaskWithRequest:request completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
     // 默认存储到临时文件夹 tmp 中,需要剪切文件到 cache
     NSLog(@"目标位置%@", location);
     NSString *fullPath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:response.suggestedFilename];
     // fileURLWithPath:有协议头; URLWithString:无协议头
     [[NSFileManager defaultManager] moveItemAtURL:location toURL:[NSURL fileURLWithPath:fullPath] error:nil];
 }];
 //发送请求
 [downTask resume];
  • NSURLSessionDataTask和NSURLSessionDownloadTask下载对比
  • DataTask可以实现离线断点下载,但是代码相对复杂
  • DownloadTask可断点下载但不能离线断点下载,边接收边写入沙盒解决下载大文件内存飙升问题

NSURLConnection

  • first deprecated in iOS 9.0
  • 文件上传

FileUpload.h

#import <Foundation/Foundation.h>

@interface FileUpload : NSObject

@end

FileUpload.m

// iOS post方式上传文件

#import "FileUpload.h"

@implementation FileUpload

+ (void)uploadFiles:(NSArray *)files msgId:(NSString *)msgId obj:(id)obj userid:(NSString *)userid {
    // 分界线的标识符
    for (NSMutableDictionary *filedic in files) {
        NSString *TWITTERFON_FORM_BOUNDARY = @"AaB03x";
        // 根据url初始化request
        NSString* URL = [NSString stringWithFormat:@"http://%@%@",NSLocalizedString(@"MQTT_IP", @""),NSLocalizedString(@"im_uploadfileURL", @"")];
        NSMutableURLRequest* request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:URL]
                                                               cachePolicy:NSURLRequestReloadIgnoringLocalCacheData
                                                           timeoutInterval:10];
        // 分界线 --AaB03x
        NSString *MPboundary=[[NSString alloc]initWithFormat:@"--%@",TWITTERFON_FORM_BOUNDARY];
        // 结束符 AaB03x--
        NSString *endMPboundary=[[NSString alloc]initWithFormat:@"%@--",MPboundary];
        // 要上传的文件
        NSData *data = [NSData dataWithContentsOfFile:[filedic objectForKey:@"filepath"]];
        
        // http body的字符串
        NSMutableString *body=[[NSMutableString alloc]init];
        // 参数的集合普通的key-value参数
        
        body = [self setParamsKey:@"uptype" value:@"1" body:body];
        body = [self setParamsKey:@"sid" value:msgId body:body];
        body = [self setParamsKey:@"uid" value:userid body:body];
        
        // 添加分界线,换行
        [body appendFormat:@"%@\r\n",MPboundary];
        // 声明文件字段,文件名
        [body appendFormat:@"Content-Disposition: form-data; name=\"upfile\"; filename=\"%@\"\r\n",[filedic objectForKey:@"serverfilename"]];
        // 声明上传文件的格式 
        [body appendFormat:@"Content-Type: %@\r\n\r\n",[self GetContentType:[filedic objectForKey:@"serverfilename"]]];
        
        // 声明结束符:--AaB03x--
        NSString *end=[[NSString alloc]initWithFormat:@"\r\n%@",endMPboundary];
        // 声明myRequestData,用来放入http body
        NSMutableData *myRequestData=[NSMutableData data];
        // 将body字符串转化为UTF8格式的二进制
        [myRequestData appendData:[body dataUsingEncoding:NSUTF8StringEncoding]];
        // 将file的data加入
        [myRequestData appendData:data];
        // 加入结束符--AaB03x--
        [myRequestData appendData:[end dataUsingEncoding:NSUTF8StringEncoding]];
        
        // 设置HTTPHeader中Content-Type的值
        NSString *content=[[NSString alloc]initWithFormat:@"multipart/form-data; boundary=%@",TWITTERFON_FORM_BOUNDARY];
        // 设置HTTPHeader
        [request setValue:content forHTTPHeaderField:@"Content-Type"];
        // 设置Content-Length
        [request setValue:[NSString stringWithFormat:@"%lu", [myRequestData length]] forHTTPHeaderField:@"Content-Length"];
        // 设置http body
        [request setHTTPBody:myRequestData];
        // http method
        [request setHTTPMethod:@"POST"];
        
        // 开线程下载
        dispatch_queue_t defaultQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
        dispatch_async(defaultQueue, ^{
            // 另开线程
            NSData *returnData = [NSURLConnection sendSynchronousRequest:request returningResponse:nil error:nil];
            NSString *returnString = [[NSString alloc] initWithData:returnData encoding:NSUTF8StringEncoding];
            NSLog(@"上传状态返回值: %@", returnString);
            //                   if ([returnString isEqualToString:@"0"]) {
            //                       [self responseLoadFinish:files msgId:msgId upload:upload];
            //                } else {
            //                        [self responseLoadFail:files msgId:msgId upload:upload];
            //                    }
        });
    }
}

+ (NSMutableString*)setParamsKey:(NSString*)key value:(NSString*)value body:(NSMutableString*)body {
    NSString *TWITTERFON_FORM_BOUNDARY = @"AaB03x";
    // 分界线 --AaB03x
    NSString *MPboundary=[[NSString alloc]initWithFormat:@"--%@",TWITTERFON_FORM_BOUNDARY];
    // 添加分界线,换行
    [body appendFormat:@"%@\r\n",MPboundary];
    // 添加字段名称,换2行
    [body appendFormat:@"Content-Disposition: form-data; name=\"%@\"\r\n\r\n",key];
    // 添加字段的值
    [body appendFormat:@"%@\r\n",value];
    return body;
}
+ (NSString*)GetContentType:(NSString*)filename {
    if ([filename hasSuffix:@".avi"]) {
        return @"video/avi";
    }  else if ([filename hasSuffix:@".bmp"]) {
        return @"application/x-bmp";
    } else if ([filename hasSuffix:@"jpeg"]) {
        return @"image/jpeg";
    } else if ([filename hasSuffix:@"jpg"]) {
        return @"image/jpeg";
    } else if ([filename hasSuffix:@"png"]) {
        // chrome/firefox 标准image/png,IE 标准image/x-png
        return @"image/png"; 
    } else if ([filename hasSuffix:@"mp3"]) {
        return @"audio/mp3";
    } else if ([filename hasSuffix:@"mp4"]) {
        return @"video/mpeg4";
    } else if ([filename hasSuffix:@"rmvb"]) {
        return @"application/vnd.rn-realmedia-vbr";
    } else if ([filename hasSuffix:@"txt"]) {
        return @"text/plain";
    } else if ([filename hasSuffix:@"xsl"]) {
        return @"application/x-xls";
    } else if ([filename hasSuffix:@"xslx"]) {
        return @"application/x-xls";
    } else if ([filename hasSuffix:@"xwd"]) {
        return @"application/x-xwd";
    } else if ([filename hasSuffix:@"doc"]) {
        return @"application/msword";
    } else if ([filename hasSuffix:@"docx"]) {
        return @"application/msword";
    } else if ([filename hasSuffix:@"ppt"]) {
        return @"application/x-ppt";
    } else if ([filename hasSuffix:@"pdf"]) {
        return @"application/pdf";
    }
    return nil;
}
@end

原生方式监听网络状态

方法一 :command + shift + 0打开 Documentation And API reference 搜索 Reachability
方法二:到网页[下载]https://developer.apple.com/library/content/samplecode/Reachability/Introduction/Intro.html

  • 获取当前网络状态
//  [Reachability reachabilityForInternetConnection];只能判断wifi是否可用,不能判断是否连接以太网络
Reachability *reachability = [Reachability reachabilityWithHostName:@"www.baidu.com"];
NetworkStatus netStatus = [reachability currentReachabilityStatus];
    switch (netStatus) {
        case NotReachable:
            break;
        case ReachableViaWiFi:
            networkStatus = 
            break;
        case ReachableViaWWAN:
            break;

        default:
            break;
    }
  • 监听网络状态
- (void) addNotifacation
{
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(reachabilityChanged:) name:kReachabilityChangedNotification object:nil];
 [self.hostReachability startNotifier];
}

- (void) reachabilityChanged:(NSNotification *)note
{
 Reachability* curReach = [note object];
 NSParameterAssert([curReach isKindOfClass:[Reachability class]]);
 NetworkStatus netStatus = [reachability currentReachabilityStatus];
}

AFNetworking 中封装的监听网络

// 1.获得网络监控的管理者
AFNetworkReachabilityManager *mgr = [AFNetworkReachabilityManager sharedManager];
// 2.设置网络状态改变后的处理
[mgr setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {
// 当网络状态改变了, 就会调用这个block
switch(status) {
    case AFNetworkReachabilityStatusUnknown: // 未知网络
              NSLog(@"未知网络");
              break;
     case AFNetworkReachabilityStatusNotReachable: // 没有网络(断网)
              NSLog(@"没有网络(断网)");
              break;
    case AFNetworkReachabilityStatusReachableViaWWAN: // 手机自带网络
              NSLog(@"手机自带网络");
              break;
    case AFNetworkReachabilityStatusReachableViaWiFi: // WIFI
              NSLog(@"WIFI");
              break;
    }
}];
// 3.开始监控
[mgr startMonitoring];

AFN和ASI有啥不同

1 AFN基于NSURL(NSURLSession&NSURLConnection),ASI基于底层的CFNetwork框架,因此ASI的性能优于AFN
2 AFN采取block的方式处理请求,ASI最初采取delegate的方式处理请求,后面也增加了block的方式
3 AFN只封装了一些常用功能,满足基本需求,直接忽略了很多扩展功能,比如没有封装同步请求;ASI提供的功能较多,预留了各种接口和工具供开发者自行扩展
4 AFN直接解析服务器返回的JSON、XML等数据,而ASI比较原始,返回的是NSData二进制数据
5 AFN在iOS9.0之后需要网络权限,而ASI不需要

在真正的开发中

  1. 如果是普通的GET&POST请求、小文件上传,强烈建议用AFN,因为AFN简单好用。

  2. 如果是下载强烈建议用ASI,因为它提供了很强大的功能。

附:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容