zoukankan      html  css  js  c++  java
  • iOS开发——网络篇——NSURLSession,下载、上传代理方法,利用NSURLSession断点下载,AFN基本使用,网络检测,NSURLConnection补充

    一、NSURLConnection补充


    前面提到的NSURLConnection有些知识点需要补充

    NSURLConnectionDataDelegate的代理方法有一下几个

    - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
    {
    
    }
    
    - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
    {
    
    }
    
    - (void)connectionDidFinishLoading:(NSURLConnection *)connection
    {
    
    }
    
    - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
    {
        
    }

     我们可以监听NSURLConnection的回调代理方法是在主线程中执行还是在子线程中执行?

     默认情况下是在主线程中执行的

    注意:

    只要利用NSURLConnection发送一个请求, 那么系统会就自动将NSURLConnection加入到当前线程的RunLoop中

    如果是在主线程中发送求情, 那么主线程的RunLoop永远存在, 所以NSURLConnection不会被释放

    如果是在子线程中发送请求, 那么子线程默认没有RunLoop, 所以NSURLConnection会被释放

    NSURLConnection在子线程发送请求时要手动添加到runloop中,否则局部变量会被释放

    start方法会自动添加到runloop,如果没有会自动创建runloop,那么系统内部会自动创建一个

    二、NSURLSession


    1、NSURLSession
    利用NSURLSession发送请求的使用步骤
    - 创建NSURLSession
    - 利用NSURLSession创建Task
    - 执行Task


    比如:

    // 1.创建Session
    NSURLSession *session = [NSURLSession sharedSession];
    // 2.根据Session创建Task
    NSURLSessionDownloadTask *task = [session downloadTaskWithRequest:request completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
    
    }];
    // 3.执行Task
    [task resume];

    Task的类型

    2、获得NSURLSession

    // 获得共享的Session
    + (NSURLSession *)sharedSession;
    
    // 自定义Session
    + (NSURLSession *)sessionWithConfiguration:(NSURLSessionConfiguration *)configuration;
    
    // 第一个参数:Session的配置信息
    // 第二个参数: 代理
    // 第三个参数: 决定了代理方法在哪个线程中执行
    + (NSURLSession *)sessionWithConfiguration:(NSURLSessionConfiguration *)configuration delegate:(id <NSURLSessionDelegate>)delegate delegateQueue:(NSOperationQueue *)queue;

    3、NSURLSessionTask常见方法

    // 暂停
    [self.task suspend];
    // 继续
    [self.task resume];
    
    // 取消
    // 注意点: 任务一旦取消, 就不能恢复了
    [self.task cancel];
    
    // 如果是调用cancelByProducingResumeData方法, 方法内部会回调一个block, 在block中会将resumeData传递给我们
    // resumeData中就保存了当前下载任务的配置信息(下载到什么地方, 从什么地方恢复等等)
    [self.task cancelByProducingResumeData:^(NSData *resumeData) {
    self.resumeData = resumeData;
    }];
    
    @property (readonly, copy) NSError *error; // 错误
    @property (readonly, copy) NSURLResponse *response; // 响应

    三、NSURLSession相关方法


    1、NSURLSessionDataTask

    // data: 服务器返回给我们的数据
    // response : 响应头
    // error: 错误信息
    - (NSURLSessionDataTask *)dataTaskWithRequest:(NSURLRequest *)request completionHandler:(void (^)(NSData *data, NSURLResponse *response, NSError *error))completionHandler NS_CLASS_AVAILABLE(NSURLSESSION_AVAILABLE, 7_0);
    
    
    
    // 如果是通过传入url的方法创建Task, 方法内部会自动根据URL创建一个Request
    // 如果是发送Get请求, 或者不需要设置请求头信息, 那么建议使用当前方法发送请求
    
    - (NSURLSessionDataTask *)dataTaskWithURL:(NSURL *)url completionHandler:(void (^)(NSData *data, NSURLResponse *response, NSError *error))completionHandler NS_CLASS_AVAILABLE(NSURLSESSION_AVAILABLE, 7_0);

    2、NSURLSessionDownloadTask

    // location: 下载好的文件在沙盒中的位置
    // NSURLSessionDownloadTask已经默认帮我们实现了, 边下载边写入到沙盒
    - (NSURLSessionDownloadTask *)downloadTaskWithRequest:(NSURLRequest *)request completionHandler:(void (^)(NSURL *location, NSURLResponse *response, NSError *error))completionHandler NS_CLASS_AVAILABLE(NSURLSESSION_AVAILABLE, 7_0);
    
    - (NSURLSessionDownloadTask *)downloadTaskWithURL:(NSURL *)url completionHandler:(void (^)(NSURL *location, NSURLResponse *response, NSError *error))completionHandler NS_CLASS_AVAILABLE(NSURLSESSION_AVAILABLE, 7_0);
    
    - (NSURLSessionDownloadTask *)downloadTaskWithResumeData:(NSData *)resumeData completionHandler:(void (^)(NSURL *location, NSURLResponse *response, NSError *error))completionHandler NS_CLASS_AVAILABLE(NSURLSESSION_AVAILABLE, 7_0);

    3、NSURLSessionUploadTask
    利用NSURLSessionUploadTask上传的步骤(和NSURLConnection类似)

    3.1.创建URL
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/upload"];
    3.2.根据URL创建NSURLRequest
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
      3.2.1设置请求头
    request.HTTPMethod = @"POST";
    [request setValue:@"multipart/form-data; boundary=分割线" forHTTPHeaderField:@"Content-Type"];
    
    3.2.2设置请求体
    NSMutableData *data = [NSMutableData data];
      2.2.1设置文件参数
      2.2.2设置非文件参数
      2.2.3设置结束符号
    request.HTTPBody = data;
    
    3.3.利用NSURLConnetion发送请求
    // 注意点: The body stream and body data in this request object are ignored.
    // 也就是说, 如果利用NSURLSessionUploadTask上传文件, 那么请求体必须写在fromData参数中, 不能设置在request中
    // 如果设置在request中会被忽略
    // request.HTTPBody = body;
    NSURLSession *session = [NSURLSession sharedSession];
    NSURLSessionUploadTask *task = [session uploadTaskWithRequest:request fromData:body completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
    
    }];
    // 第一个参数: 需要请求的地址/请求头/请求体
    // 第二个参数: 需要上传文件的URL
    // 注意:fromFile方法是用于PUT请求上传文件的
    // 而我们的服务器只支持POST请求上传文件
    - (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request fromFile:(NSURL *)fileURL completionHandler:(void (^)(NSData *data, NSURLResponse *response, NSError *error))completionHandler NS_CLASS_AVAILABLE(NSURLSESSION_AVAILABLE, 7_0);
    
    
    - (NSURLSessionUploadTask *)uploadTaskWithRequest:(NSURLRequest *)request fromData:(NSData *)bodyData completionHandler:(void (^)(NSData *data, NSURLResponse *response, NSError *error))completionHandler NS_CLASS_AVAILABLE(NSURLSESSION_AVAILABLE, 7_0);

    四、Data、下载、上传代理方法


    1、NSURLSessionDataDelegate

    // 接收到服务器的响应时调用
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler
    {
    // 注意: 在NSURLSessionDataTask的代理方法中, 默认情况下是不接受服务器返回的数据的, 所以didReceiveData方法和didCompleteWithError默认不会被调用
    // 如果想接收服务器返回的数据, 必须手动的告诉系统, 我们需要接收数据
    
    // void (^)(NSURLSessionResponseDisposition)
    // 返回值 (block)(参数)
    completionHandler(NSURLSessionResponseAllow);
    }
    // 接收到服务器返回的数据时调用
    // data 此次接收到的数据
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
    {
    }
    
    // 请求完毕时调用, 如果error有值, 代表请求错误
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
    {
    }

    2、NSURLSessionDownloadDelegate

    // 接收到服务器返回的数据时调用
    // 该方法会调用一次或多次
    // didWriteData : 此次接收到的数据大小
    // totalBytesWritten : 总共接收到的数据大小
    // totalBytesExpectedToWrite : 服务器总共会返回给我们的文件大小
    - (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite
    {
    }
    // 写入完成时调用
    - (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didFinishDownloadingToURL:(NSURL *)location
    {
    }
    // 恢复下载时调用
    - (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didResumeAtOffset:(int64_t)fileOffset expectedTotalBytes:(int64_t)expectedTotalBytes
    {
    }
    
    // 下载完成时调用
    // 如果调用改方法时error有值, 代表着下载出现错误
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task
    didCompleteWithError:(NSError *)error
    {
    }

    3、NSURLSessionTaskDelegate

    // 上传过程中调用
    // bytesSent: 当前这一次上传的数据大小
    // totalBytesSent: 总共已经上传的数据大小
    // totalBytesExpectedToSend: 需要上传文件的大小
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didSendBodyData:(int64_t)bytesSent totalBytesSent:(int64_t)totalBytesSent totalBytesExpectedToSend:(int64_t)totalBytesExpectedToSend
    {
        // 计算上传进度
        NSLog(@"%f", 1.0 * totalBytesSent / totalBytesExpectedToSend);
    }
    
    // 请求完毕时调用
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
    {
    }

    五、利用NSURLSession断点下载


    #import "ViewController.h"
    #import "NSString+NJ.h"
    
    @interface ViewController ()<NSURLSessionDataDelegate>
    
    @property (weak, nonatomic) IBOutlet UIProgressView *progressView;
    
    - (IBAction)start:(UIButton *)sender;
    
    - (IBAction)pause:(UIButton *)sender;
    
    - (IBAction)goOn:(UIButton *)sender;
    
    @property (nonatomic, assign)NSUInteger totalLength; /**< 总大小 */
    @property (nonatomic, assign)NSUInteger currentLength; /**< 当前已经下载的大小 */
    
    @property (nonatomic, strong) NSOutputStream *outputStream ; /**< 输出流 */
    
    @property (nonatomic, strong) NSURLSession *session; /**< session */
    
    @property (nonatomic, strong) NSURLSessionDataTask *task; /**< 任务 */
    
    @property (nonatomic, copy) NSString *path; /**< 文件路径 */
    
    @end
    
    @implementation ViewController
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
        // 初始化操作
        // 1.初始化文件路径
        self.path = [@"minion_02.mp4" cacheDir];
        NSLog(@"%@",self.path);
        // 2.初始化当前下载进度
        self.currentLength = [self getFileSizeWithPath:self.path];
    }
    
    - (IBAction)start:(UIButton *)sender
    {
        // 3.执行Task
        [self.task resume];
    }
    
    - (IBAction)pause:(UIButton *)sender
    {
        [self.task suspend];
    }
    
    - (IBAction)goOn:(UIButton *)sender
    {
        [self.task resume];
    }
    
    // 根据路径获取文件大小
    - (NSUInteger)getFileSizeWithPath:(NSString *)path
    {
        NSUInteger currentSize = [[[NSFileManager defaultManager] attributesOfItemAtPath:path error:nil][NSFileSize] integerValue];
        return currentSize;
    }
    
    #pragma mark - NSURLSessionDataDelegate
    // 接收到服务器的响应时调用
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler
    {
        NSLog(@"didReceiveResponse");
        // 告诉系统需要接收数据
        completionHandler(NSURLSessionResponseAllow);
        
        // 初始化文件总大小
        self.totalLength = response.expectedContentLength + [self getFileSizeWithPath:self.path];
        
        // 打开输出流
        self.outputStream = [NSOutputStream outputStreamToFileAtPath:self.path append:YES];
        [self.outputStream open];
    }
    
    // 接收到服务器返回的数据时调用
    // data 此次接收到的数据
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
    {
    //    NSLog(@"didReceiveData");
        // 累加已经下载的大小
        self.currentLength += data.length;
        
        // 计算进度
        self.progressView.progress = 1.0 * self.currentLength / self.totalLength;
        
        // 写入数据
        [self.outputStream write:data.bytes maxLength:data.length];
    }
    
    // 请求完毕时调用, 如果error有值, 代表请求错误
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
    {
        NSLog(@"didCompleteWithError");
        
        // 关闭输出流
        [self.outputStream close];
    }
    
    #pragma mark - lazy
    - (NSURLSession *)session
    {
        if (!_session) {
    
             // 1.创建Session
            _session =  [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
        }
        return _session;
    }
    
    - (NSURLSessionDataTask *)task
    {
        if (!_task) {
            NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/resources/videos/minion_02.mp4"];
            NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
            // 设置请求头
            NSString *range = [NSString stringWithFormat:@"bytes:%zd-", [self getFileSizeWithPath:self.path]];
            [request setValue:range forHTTPHeaderField:@"Range"];
            
            _task = [self.session dataTaskWithRequest:request];
        }
        return _task;
    }
    @end

      

    六、AFN基本使用


    1、AFN简介
    什么是AFN
    全称是AFNetworking,是对NSURLConnection的一层封装
    虽然运行效率没有ASI高,但是使用比ASI简单
    在iOS开发中,使用比较广泛
     
    AFN的github地址
     
    AFN的版本选择
     
    2、AFN对NSURLConnection封装
    GET
    // 1.创建AFN管理者
        AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
        
        // 2.利用AFN管理者发送请求
        /*
         第1个参数: 需要请求的URL地址字符串
         第2个参数: 请求时需要传递的参数
         第3个参数: 请求成功的回调
         第4个参数: 请求失败的回调
         */
        // 注意: 字典参数中不需要写? , AFN会自动添加
        NSDictionary *dict = @{
                               @"username":@"520it",
                               @"pwd":@"520it",
                               @"type":@"JSON",
                               };
        [manager GET:@"http://120.25.226.186:32812/login" parameters:dict success:^(AFHTTPRequestOperation *operation, id responseObject) {
    
            // responseObject 服务器返回的数据
            // 注意点: 如果服务器返回的是JSON, AFN会自动转换为OC对象
            NSLog(@"%@", responseObject);
    
        } failure:^(AFHTTPRequestOperation *operation, NSError *error) {
            NSLog(@"error");
        }];

    POST

    // 1.创建AFN管理者
        AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
        
        // 2.利用AFN管理者发送请求
        NSDictionary *dict = @{
                               @"username":@"520it",
                               @"pwd":@"520it",
                               @"type":@"JSON",
                               };
        [manager POST:@"http://120.25.226.186:32812/login" parameters:dict success:^(AFHTTPRequestOperation *operation, id responseObject) {
            NSLog(@"%@", responseObject);
        } failure:^(AFHTTPRequestOperation *operation, NSError *error) {
            NSLog(@"error");
        }];

    3、AFN对NSURLSession封装

    GET

        // 1.创建AFN管理者
        AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
        // 2.利用AFN发送请求
        NSDictionary *dict = @{
                               @"username":@"520it",
                               @"pwd":@"520it",
                               @"type":@"JSON",
                               };
        [manager GET:@"http://120.25.226.186:32812/login" parameters:dict success:^(NSURLSessionDataTask *task, id responseObject) {
            NSLog(@"%@", responseObject);
        } failure:^(NSURLSessionDataTask *task, NSError *error) {
            NSLog(@"error");
        }];

    POST

        // 1.创建AFN管理者
        AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
        // 2.利用AFN发送请求
        NSDictionary *dict = @{
                               @"username":@"520it",
                               @"pwd":@"520it",
                               @"type":@"JSON",
                               };
        [manager POST:@"http://120.25.226.186:32812/login" parameters:dict success:^(NSURLSessionDataTask *task, id responseObject) {
            NSLog(@"%@", responseObject);
        } failure:^(NSURLSessionDataTask *task, NSError *error) {
            NSLog(@"error");
        }];

    4、使用AFN下载文件

        // 1.创建AFN管理者
        AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
        // 2.利用AFN下载文件
        NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://120.25.226.186:32812/resources/images/minion_02.png"]];
        NSURLSessionDownloadTask *task = [manager downloadTaskWithRequest:request progress:nil destination:^NSURL *(NSURL *targetPath, NSURLResponse *response) {
            // 请求完成的回调
            // targetPath : 下载好的文件的路径
            // response: 响应头
            // block的返回值: 告诉AFN框架, 是否需要将下载好的文件移动到其它路径下
            NSLog(@"targetPath = %@", targetPath);
            NSString *path = [response.suggestedFilename cacheDir];
            NSURL *destURL = [NSURL fileURLWithPath:path];
            NSLog(@"destURL = %@", destURL);
            return destURL;
        } completionHandler:^(NSURLResponse *response, NSURL *filePath, NSError *error) {
            // 下载完成的回调
            // filePath : 移动之后的文件路径
            NSLog(@"filePath = %@", filePath);
        }];
        
        [task resume];

    监听下载进度

    在NSProgress有两个属性

         @property int64_t totalUnitCount;  需要下载文件的总大小

         @property int64_t completedUnitCount;  已经下载文件的总大小

    使用KVO进行监听completedUnitCount值的变化

    - (void)download2
    {
        // 1.创建AFN管理者
        AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
        
        NSProgress *progress = nil;
        self.progress = progress;
        
        // 2.利用AFN下载文件
        NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://120.25.226.186:32812/resources/videos/minion_02.mp4"]];
        
        // 只要将NSProgress对象传递给AFN, AFN就会在内部自动给NSProgress赋值 会自动修改NSProgress的下载进度
        NSURLSessionDownloadTask *task = [manager downloadTaskWithRequest:request progress:&progress destination:^NSURL *(NSURL *targetPath, NSURLResponse *response) {
            // 请求完成的回调
            NSString *path = [response.suggestedFilename cacheDir];
            NSURL *destURL = [NSURL fileURLWithPath:path];
            return destURL;
        } completionHandler:^(NSURLResponse *response, NSURL *filePath, NSError *error) {
            // 下载完成的回调
            // filePath : 移动之后的文件路径
            NSLog(@"filePath = %@", filePath);
        }];
        
        // 给NSProgress注册监听, 监听它的completedUnitCount属性的改变
        [progress addObserver:self forKeyPath:@"completedUnitCount" options:NSKeyValueObservingOptionNew context:nil];
        
        // 注意: 必须手动执行才会开始下载
        [task resume];
        
    }
    - (void)dealloc { [self removeObserver:self.progress forKeyPath:@"completedUnitCount"]; }
    - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { if ([object isKindOfClass:[NSProgress class]]) { NSProgress *progress = (NSProgress *)object; // 计算下载进度 NSLog(@"%f", 1.0 * progress.completedUnitCount / progress.totalUnitCount); } }

    5、AFN上传文件

     // 1.创建AFN管理者
        AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
        
        // 2.利用AFN上传文件
        NSDictionary *dict = @{
                               @"username":@"lnj"
                               };
        [manager POST:@"http://120.25.226.186:32812/upload" parameters:dict constructingBodyWithBlock:^(id<AFMultipartFormData> formData) {
            // 在这个block中上传文件数据
            // formData就是专门用于保存需要上传文件的二进制数据的
    
            NSData *data = [NSData dataWithContentsOfFile:@"/Users/chg/Desktop/胸小别讲话.jpg"];
            // 注意: 该方法不能用于POST上传
    //        [formData appendPartWithFormData:data name:@"file"];
            /*
             第一个参数: 需要上传的文件二进制
             第二个参数: 服务器对应的参数名称
             第三个参数: 文件的名称
             第四个参数: 文件的MIME类型
             */
    //        [formData appendPartWithFileData:data name:@"file" fileName:@"abc.png" mimeType:@"image/png"];
            
            /*
             第一个参数: 需要上传的文件的URL
             第二个参数: 服务器对应的参数名称
             第三个参数: 文件的名称
             第四个参数: 文件的MIME类型
             */
            NSURL *url = [NSURL fileURLWithPath:@"/Users/chg/Desktop/胸小别讲话.jpg"];
    //        [formData appendPartWithFileURL:url name:@"file" fileName:@"def.png" mimeType:@"image/png" error:nil];
            
            // 如果使用以下方法上传文件, AFN会自动获取文件的名称和类型
            [formData appendPartWithFileURL:url name:@"file" error:nil];
            
        } success:^(NSURLSessionDataTask *task, id responseObject) {
            NSLog(@"成功的回调 %@", responseObject);
        } failure:^(NSURLSessionDataTask *task, NSError *error) {
            NSLog(@"失败的回调");
        }];

    6、AFN序列化

        // 1.创建AFN管理者
        AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
        
        // 如果服务器返回的不是JSON数据, 那么需要提前告诉AFN如何处理处理数据
        
        // 只要给AFN的responseSerializer属性, 赋值为AFXMLParserResponseSerializer, 以后AFN就会将服务器返回的数据当做XML来处理
    //    manager.responseSerializer = [AFXMLParserResponseSerializer serializer];
        
        // 只要给AFN的responseSerializer属性, 赋值为AFHTTPResponseSerializer, 以后AFN就会将服务器返回的数据当做二进制来处理, 也就是说不做任何处理
    //    manager.responseSerializer = [AFHTTPResponseSerializer serializer];
        
        
        // 2.利用AFN发送请求
        NSDictionary *dict = @{
                               @"username":@"520it",
                               @"pwd":@"520it",
                               @"type":@"XML",
                               };
        
        [manager POST:@"http://120.25.226.186:32812/resources/images/minion_02.png" parameters:nil success:^(NSURLSessionDataTask *task, id responseObject) {
            // 注意: AFN默认会将服务器返回的数据当做JSON处理
            // 如果服务器返回的是JSON数据, 那么会自动转换为OC对象
            NSLog(@"%@", responseObject);
            
            // 只要设置AFN的responseSerializer为XML, 那么返回的responseObject就是NSXMLParser解析器
            
        } failure:^(NSURLSessionDataTask *task, NSError *error) {
            NSLog(@"error");
        }];

    7、网络检测

        // 1.创建网络监听管理者
        AFNetworkReachabilityManager *manager = [AFNetworkReachabilityManager sharedManager];
        
        // 2.设置网络变化的回调
        [manager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {
            // 只要用户的网络发生改变, 就会调用这个block
            /*
             AFNetworkReachabilityStatusUnknown          = 不能识别,
             AFNetworkReachabilityStatusNotReachable     = 没有网络,
             AFNetworkReachabilityStatusReachableViaWWAN = 蜂窝网,
             AFNetworkReachabilityStatusReachableViaWiFi = 局域网,
             */
            switch (status) {
                case AFNetworkReachabilityStatusReachableViaWWAN:
                    NSLog(@"蜂窝网");
                    break;
                case AFNetworkReachabilityStatusReachableViaWiFi:
                    NSLog(@"局域网");
                    break;
                case AFNetworkReachabilityStatusNotReachable:
                    NSLog(@"没有网络");
                    break;
                default:
                    NSLog(@"不能识别");
                    break;
            }
        }];
        
        // 3.开始监听
        [manager startMonitoring];

    苹果的有个类(Reachability)用于检测网络

    这个类需要自己下载

    如果要检测当前网络可以这样实现

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        // 1.创建Reachability对象
        self.r1 = [Reachability reachabilityForLocalWiFi];
        // 2.给Reachability对象注册通知
        [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(getNetworkStatus) name:kReachabilityChangedNotification object:nil];
        // 3.让Reachability对象发送网络改变的通知
        [self.r1 startNotifier];
        
    }
    
    - (void)dealloc
    {
        [[NSNotificationCenter defaultCenter] removeObserver:self];
    }
    
    - (void)getNetworkStatus
    {
        if ([Reachability reachabilityForLocalWiFi].currentReachabilityStatus != NotReachable) {
            NSLog(@"当前是WiFi");
        }else if ([Reachability reachabilityForInternetConnection].currentReachabilityStatus != NotReachable)
        {
            NSLog(@"当前是蜂窝网");
        }else
        {
            NSLog(@"没有网络");
        }
    }

    8、总结

    总的来说AFN框架分为5大块,通过下面这张图可以一目了然

    通过cocoapods更新该框架也可以看到作者已经为这个框架的类用文件夹分好了,有助于我们阅读

    将来的你会感谢今天如此努力的你! 版权声明:本文为博主原创文章,未经博主允许不得转载。
  • 相关阅读:
    Shiro 登录、退出、校验是否登录涉及到的Session和Cookie
    Apache Tomcat 8.0 官方文档
    FastDFS分布式文件系统(主备Tracker、主备Storage)
    PHP文件系统
    PHP 文件包含
    PHP函数
    PHP 全局变量
    PHP7新增知识点
    PHP数据
    PHP常量
  • 原文地址:https://www.cnblogs.com/chglog/p/4755682.html
Copyright © 2011-2022 走看看