iOS開髮網絡篇—發送GET和POST請求(使用NSURLSession) - 轉

来源:https://www.cnblogs.com/jiuyi/archive/2018/12/13/10114450.html
-Advertisement-
Play Games

說明: 1.該文主要介紹如何使用NSURLSession來發送GET請求和POST請求 2.本文將不再講解NSURLConnection的使用,如有需要瞭解NSURLConnection如何發送請求。 詳細信息,請參考:http://www.cnblogs.com/wendingding/p/381 ...


說明:

  1.該文主要介紹如何使用NSURLSession來發送GET請求和POST請求

  2.本文將不再講解NSURLConnection的使用,如有需要瞭解NSURLConnection如何發送請求。

    詳細信息,請參考:http://www.cnblogs.com/wendingding/p/3813706.html

  3.本文示例代碼發送的請求均為http請求,已經對info.plist文件進行配置。

    如何配置,請參考:https://github.com/HanGangAndHanMeimei/iOS9AdaptationTips

  4.本文示例代碼,可以在下麵的地址獲取:

    https://github.com/HanGangAndHanMeimei/Code

一、簡單說明

  在iOS9.0之後,以前使用的NSURLConnection過期,蘋果推薦使用NSURLSession來替換NSURLConnection完成網路請求相關操作。

  NSURLSession的使用非常簡單,先根據會話對象創建一個請求Task,然後執行該Task即可。

  NSURLSessionTask本身是一個抽象類,在使用的時候,通常是根據具體的需求使用它的幾個子類。關係如下:

二、發送GET請求

  使用NSURLSession發送GET請求的方法和NSURLConnection類似,整個過程如下:

    1.確定請求路徑(一般由公司的後臺開發人員以介面文檔的方式提供),GET請求參數直接跟在URL後面

    2.創建請求對象(預設包含了請求頭和請求方法【GET】),此步驟可以省略

    3.創建會話對象(NSURLSession)

    4.根據會話對象創建請求任務(NSURLSessionDataTask)

    5.執行Task

    6.當得到伺服器返回的響應後,解析數據(XML|JSON|HTTP)

  示例代碼:

-(void)get1
{
    //對請求路徑的說明
    //http://120.25.226.186:32812/login?username=520it&pwd=520&type=JSON
    //協議頭+主機地址+介面名稱+?+參數1&參數2&參數3
    //協議頭(http://)+主機地址(120.25.226.186:32812)+介面名稱(login)+?+參數1(username=520it)&參數2(pwd=520)&參數3(type=JSON)
    //GET請求,直接把請求參數跟在URL的後面以?隔開,多個參數之間以&符號拼接
    
    //1.確定請求路徑
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=520it&pwd=520it&type=JSON"];
    
    //2.創建請求對象
    //請求對象內部預設已經包含了請求頭和請求方法(GET)
    NSURLRequest *request = [NSURLRequest requestWithURL:url];
    
    //3.獲得會話對象
    NSURLSession *session = [NSURLSession sharedSession];
      
    //4.根據會話對象創建一個Task(發送請求)
    /*
     第一個參數:請求對象
     第二個參數:completionHandler回調(請求完成【成功|失敗】的回調)
               data:響應體信息(期望的數據)
               response:響應頭信息,主要是對伺服器端的描述
               error:錯誤信息,如果請求失敗,則error有值
     */
    NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
        
        if (error == nil) {
            //6.解析伺服器返回的數據
            //說明:(此處返回的數據是JSON格式的,因此使用NSJSONSerialization進行反序列化處理)
            NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil];
            
            NSLog(@"%@",dict);
        }
    }];
    
    //5.執行任務
    [dataTask resume];
}

發送GET請求的第一種方法
View Code

 

-(void)get2
{
    //1.確定請求路徑
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=520it&pwd=520it&type=JSON"];
    
    //2.獲得會話對象
    NSURLSession *session = [NSURLSession sharedSession];
    
    //3.根據會話對象創建一個Task(發送請求)
    /*
     第一個參數:請求路徑
     第二個參數:completionHandler回調(請求完成【成功|失敗】的回調)
               data:響應體信息(期望的數據)
               response:響應頭信息,主要是對伺服器端的描述
               error:錯誤信息,如果請求失敗,則error有值
     註意:
        1)該方法內部會自動將請求路徑包裝成一個請求對象,該請求對象預設包含了請求頭信息和請求方法(GET)
        2)如果要發送的是POST請求,則不能使用該方法
     */
    NSURLSessionDataTask *dataTask = [session dataTaskWithURL:url completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
        
        //5.解析數據
        NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil];
        NSLog(@"%@",dict);
        
    }];
    
    //4.執行任務
    [dataTask resume];
}

發送GET請求的第二種方法

    執行結果
![](http://images2015.cnblogs.com/blog/450136/201601/450136-20160129131707286-1497362695.png)
View Code

 

此處列印的值是一個字典,字典中success這個key對應的value列印出來為Unicode編碼的,如果想輸出中文,可以為NSDictionary提供一個分類,重寫系統中的方法


#import "NSDictionary+Log.h"

@implementation NSDictionary (Log)

-(NSString *)descriptionWithLocale:(id)locale indent:(NSUInteger)level
{
    //初始化可變字元串
    NSMutableString *string = [NSMutableString string];
    //拼接開頭[
    [string appendString:@"["];
    
    //拼接字典中所有的鍵值對
    [self enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
        [string appendFormat:@"%@:",key];
        [string appendFormat:@"%@",obj];
    }];
    
    //拼接結尾]
    [string appendString:@"]"];
    
    return string;
}

@end

字典分類中重寫系統方法
View Code

執行結果:

三、發送POST請求

  使用NSURLSession發送POST請求的方法和NSURLConnection類似,整個過程如下:

    1)確定請求路徑(一般由公司的後臺開發人員以介面文檔的方式提供)

    2)創建可變的請求對象(因為需要修改),此步驟不可以省略

    3)修改請求方法為POST

    4)設置請求體,把參數轉換為二進位數據並設置請求體

    5)創建會話對象(NSURLSession)

    6)根據會話對象創建請求任務(NSURLSessionDataTask)

    7)執行Task

    8)當得到伺服器返回的響應後,解析數據(XML|JSON|HTTP)

  示例代碼:

-(void)post
{
    //對請求路徑的說明
    //http://120.25.226.186:32812/login
    //協議頭+主機地址+介面名稱
    //協議頭(http://)+主機地址(120.25.226.186:32812)+介面名稱(login)
    //POST請求需要修改請求方法為POST,並把參數轉換為二進位數據設置為請求體
    
    //1.創建會話對象
    NSURLSession *session = [NSURLSession sharedSession];
    
    //2.根據會話對象創建task
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login"];
    
    //3.創建可變的請求對象
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    
    //4.修改請求方法為POST
    request.HTTPMethod = @"POST";
    
    //5.設置請求體
    request.HTTPBody = [@"username=520it&pwd=520it&type=JSON" dataUsingEncoding:NSUTF8StringEncoding];
    
    //6.根據會話對象創建一個Task(發送請求)
    /*
     第一個參數:請求對象
     第二個參數:completionHandler回調(請求完成【成功|失敗】的回調)
                data:響應體信息(期望的數據)
                response:響應頭信息,主要是對伺服器端的描述
                error:錯誤信息,如果請求失敗,則error有值
     */
    NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
        
        //8.解析數據
        NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil];
        NSLog(@"%@",dict);
        
    }];
    
    //7.執行任務
    [dataTask resume];
}

發送POST請求的方法
View Code

 

四、NSURLSession代理方法簡單介紹

  有的時候,我們可能需要監聽網路請求的過程(如下載文件需監聽文件下載進度),那麼就需要用到代理方法。

  接下來通過代碼簡單說明NSURLSession中普通網路請求會涉及代理方法的使用

#import "ViewController.h"

@interface ViewController ()<NSURLSessionDataDelegate>
@property (nonatomic, strong) NSMutableData *responseData;
@end

@implementation ViewController

-(NSMutableData *)responseData
{
    if (_responseData == nil) {
        _responseData = [NSMutableData data];
    }
    return _responseData;
}

//當點擊控制器View的時候會調用該方法
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    [self delegateTest];
}

//發送請求,代理方法
-(void)delegateTest
{
    //1.確定請求路徑
    NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/login?username=520it&pwd=520it&type=JSON"];
    
    //2.創建請求對象
    //請求對象內部預設已經包含了請求頭和請求方法(GET)
    NSURLRequest *request = [NSURLRequest requestWithURL:url];
    
    //3.獲得會話對象,並設置代理
    /*
     第一個參數:會話對象的配置信息defaultSessionConfiguration 表示預設配置
     第二個參數:誰成為代理,此處為控制器本身即self
     第三個參數:隊列,該隊列決定代理方法在哪個線程中調用,可以傳主隊列|非主隊列
     [NSOperationQueue mainQueue]   主隊列:   代理方法在主線程中調用
     [[NSOperationQueue alloc]init] 非主隊列: 代理方法在子線程中調用
     */
    NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
    
    //4.根據會話對象創建一個Task(發送請求)
    NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request];
    
    //5.執行任務
    [dataTask resume];
}

//1.接收到伺服器響應的時候調用該方法
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler
{
    //在該方法中可以得到響應頭信息,即response
    NSLog(@"didReceiveResponse--%@",[NSThread currentThread]);
    
    //註意:需要使用completionHandler回調告訴系統應該如何處理伺服器返回的數據
    //預設是取消的
    /*
        NSURLSessionResponseCancel = 0,        預設的處理方式,取消
        NSURLSessionResponseAllow = 1,         接收伺服器返回的數據
        NSURLSessionResponseBecomeDownload = 2,變成一個下載請求
        NSURLSessionResponseBecomeStream        變成一個流
     */
    
    completionHandler(NSURLSessionResponseAllow);
}

//2.接收到伺服器返回數據的時候會調用該方法,如果數據較大那麼該方法可能會調用多次
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
{
    NSLog(@"didReceiveData--%@",[NSThread currentThread]);
    
    //拼接伺服器返回的數據
    [self.responseData appendData:data];
}

//3.當請求完成(成功|失敗)的時候會調用該方法,如果請求失敗,則error有值
-(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
    NSLog(@"didCompleteWithError--%@",[NSThread currentThread]);
    
    if(error == nil)
    {
        //解析數據,JSON解析請參考http://www.cnblogs.com/wendingding/p/3815303.html
        NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:self.responseData options:kNilOptions error:nil];
        NSLog(@"%@",dict);
    }
}
@end
View Code

執行結果:

原文轉自http://www.cnblogs.com/wendingding/p/5168772.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 創建資料庫快照註意事項 語法:set transaction isolation level snapshot; 指定事務中任何語句讀取的數據都將是在事務開始時便存在的數據的事務上一致的版本。 事務只能識別在其開始之前提交的數據修改。 在當前事務中執行的語句將看不到在當前事務開始以後由其他事務所做的 ...
  • hive中left/right join on連接中and與where的使用問題 ...
  • 1. Docker搭建Mongodb 1.1 獲取docker鏡像 1.2 創建mongodb容器 如果加需要驗證就加 auth,不需要驗證,就去掉。預設mongodb是不使用用戶認證 1.3 進入容器設置用戶 或者直接進入admin 1.4 測試 查看是否連接成功 2.維護mongoDB 2.1 ...
  • Adroid佈局 有人形象地比喻,Android開發中的佈局就相當於一棟建築的外觀架構。佈局用得好,這棟建築的外觀才美觀高大上。 Android佈局管理器 Android佈局管理器本身是一個界面控制項,所有的佈局管理器都是ViewGroup類的子類,都是可以當做容器類來使用的。因此一個佈局管理器中可以 ...
  • 轉載請標明出處,維權必究:https://www.cnblogs.com/tangZH/p/10116298.html 在項目過程中出現了上述錯誤。 會出現這樣的錯誤是在我使用: notifyItemRemoved(position); notifyItemRangeChanged(position ...
  • 轉載請標明出處,維權必究:https://www.cnblogs.com/tangZH/p/10116095.html 我們為了移除RecycleView的某一項,會用RecycleView的notifyItemRemoved(int position)方法,但是需要註意的是:1、用該方法之後並不會 ...
  • 一、思路 第一,圖片拖拽位置互換/刪除,參照第三方; 第二,圖片用scrollview瀏覽,縮放用zoomToRect,不用CGAffineTransformScale; 其次,還要返回當前縮放圖片 二、核心代碼就不貼了,HDragItemListView.m主要處理圖片拖拽的功能 三、效果圖 Gi ...
  • 導入項目時,發現之前項目的butter knife報錯,用到註解的應該都會報錯Error:Execution failed for task ':app:javaPreCompileDebug'.> Annotation processors must be explicitly declared ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...