Swift # GET&POST請求 網絡緩存的簡單處理


 GET & POST 的對比

 

源碼:https://github.com/SpongeBob-GitHub/Get-Post.git

 

 1. URL

    - GET

        所有的參數都包含在 URL 中

 

        1. 如果需要添加參數,腳本后面使用 `?`

        2. 參數格式:值對

            參數名=值

        3. 如果有多個參數,使用 `&` 連接

        4. 在 get 方法中,url字符串中不能包含中文或者特殊符號 空格

           如果出現,需要添加百分號轉義

 

    - POST

        URL中不包含參數,直接指定登錄腳本即可

 

 2. Request

 

    - GET

        - 因為 GET 的效率高,而且性能好,能夠被緩存,使用頻率高

        - 是默認的請求方法,不需要任何設定

 

    - POST

        - 需要指定請求方法

            request.HTTPMethod = @"POST";

        - 所有參數都包含在請求體中,二進制數據,來源:firebug的源代碼中粘貼

        - 參數格式,和 GET 的幾乎一致,只是沒有 `?`

        - POST方法獲得的網絡數據不能被緩存

 

 3. Connection

    這是一個最單純的網絡方法,只是發送"請求",接收服務器返回的二進制"實體數據"

 

    GET & POST 沒有任何區別

 

GET:

    /// GET 登錄
    func getLogin() {
        let username = "SpongeBob"
        let pwd = "1234567890"
        
        /**
        GET 方法中,所有的參數都包含在 URL 中
        
        提示:login.php 是測試用的腳本
        
        【需要:在Mac電腦上,配置Apache服務器--http://www.cnblogs.com/SpongeBob-GitHub/p/4496164.html & 將login.php腳本放到服務器上!】
        注意:得先啟動配置好的Apache服務器
            $ sudo apachectl -k restart
        
        1. 如果需要添加參數,使用 ?
        2. 參數格式:值對
        參數名=值
        3. 如果有多個參數,使用 & 連接
        
        4. 在 get 方法中,url字符串中不能包含中文或者特殊符號 空格
        如果出現,需要添加[百分號]轉義
        */
        var urlString = NSString(format: "http://127.0.0.1/login.php?username=\(username)&password=\(pwd)")
        // 利用UTF8編碼
        urlString = urlString.stringByAddingPercentEscapesUsingEncoding(NSUTF8StringEncoding)!
        let url = NSURL(string: urlString as String)
        
        // 因為 GET 的效率高,而且性能好,因此在網絡訪問中,使用頻率非常高!默認的請求方法就是 GET 的,無需指定!
        // GET 方法是可以緩存的!
        var request = NSURLRequest(URL: url!, cachePolicy: NSURLRequestCachePolicy.ReturnCacheDataElseLoad, timeoutInterval: 10.0)
        
        println(request.HTTPMethod)
        
        NSURLConnection.sendAsynchronousRequest(request, queue: NSOperationQueue.mainQueue()) { (response, data, connectionError) -> Void in
            
            println(response)
            
            if connectionError != nil {
                
                println("login error....\(connectionError)")

            } else {
                
                // 這里的data數據,就是通過GET請求獲取到的,第一次獲取成功后,就緩存到本地沙盒中,下次就不再獲取,直接從本地加載!!!--緩存
                
                // 問題:但是如果這時候,服務器中返回的數據修改了,這里依然會打印修改前的data!
                // 比如:將login.php腳本中得返回userName改成 xxxx,依然會打印為SpongeBob。
                
                // FIXME: 這種問題一般會出現在圖片的獲取,我們重新寫一個類來解決這個問題!
                
                var dict = NSJSONSerialization.JSONObjectWithData(data, options: NSJSONReadingOptions.MutableContainers, error: nil) as! NSDictionary
                
                println("GET: -> \(dict)")
            }
            
        }
    }

 

POST:

    /// POST 登錄
    func postLogin() {
        let username = "SpongeBob"
        let pwd = "1234567890"

        let url = NSURL(string: "http://localhost/login.php")
        var request = NSMutableURLRequest(URL: url!, cachePolicy: NSURLRequestCachePolicy.ReturnCacheDataElseLoad, timeoutInterval: 10.0)
        request.HTTPMethod = "POST"
        
        println(request.HTTPMethod)

        // 設置請求體為二進制數據
        var bodyStr = NSString(format: "username=\(username)&password=\(pwd)")
        request.HTTPBody = bodyStr.dataUsingEncoding(NSUTF8StringEncoding)
        
        NSURLConnection.sendAsynchronousRequest(request, queue: NSOperationQueue.mainQueue()) { (response, data, connectionError) -> Void in
            
            println(response)
            
            if connectionError != nil {
                
                println("login error....\(connectionError)")
                
            } else {
                
                var dict = NSJSONSerialization.JSONObjectWithData(data, options: NSJSONReadingOptions.MutableContainers, error: nil) as! NSDictionary
                
                println("POST: -> \(dict)")
            }
        }
    }

 

針對於GET請求緩存----》 

這里的data數據,就是通過GET請求獲取到的,第一次獲取成功后,就緩存到本地沙盒中,下次就不再獲取,直接從本地加載!!!--緩存

問題:但是如果這時候,服務器中返回的數據修改了,這里依然會打印修改前的data!

比如:將login.php腳本中得返回userName改成 xxxx,依然會打印為SpongeBob。

  FIXME: 這種問題一般會出現在圖片的獲取,我們重新寫一個類來解決這個問題!

 

    /// GET 加載網絡圖片 -- 緩存
    func loadImageWithGET() {
        let url = NSURL(string: "http://localhost/images/1.png")
        var request = NSMutableURLRequest(URL: url!, cachePolicy: NSURLRequestCachePolicy.ReloadIgnoringLocalAndRemoteCacheData, timeoutInterval: 10.0)
        
        // 設置請求頭 - 利用“If-None-Match”判斷是否改變
        request.setValue(self.etag, forHTTPHeaderField: "If-None-Match")
        
        println(request.HTTPMethod)
        
        NSURLConnection.sendAsynchronousRequest(request, queue: NSOperationQueue.mainQueue()) { (response, data, connectionError) -> Void in
            
            var httpRespone = response as! NSHTTPURLResponse
            println("\(httpRespone.allHeaderFields) + \(httpRespone)")
            
            // 第2次點擊時,為304,即需要從本地加載
            
            if httpRespone.statusCode == 304 {
                println("Loading the local data...")
                
                // NSURLCache
                var cacheResponse = NSURLCache.sharedURLCache().cachedResponseForRequest(request)
                self.imgView.image = UIImage(data: cacheResponse!.data)
                return
            }
            
            self.etag = httpRespone.allHeaderFields["Etag"] as? String
            
            
            self.imgView.image = UIImage(data: data)
        }
    }

 

網絡緩存的處理

/**

 * [iOS] 簡述 NSURLCache 的存儲目錄

 **

    (lldb) po NSHomeDirectory()

    "/Users/xxxx/Library/Developer/CoreSimulator/Devices/5A46B8A4-xxxx-4B6A-B5B8-F76A6E13998F/data/Containers/Data/Application/EFEF0D66-xxxx-4EE6-B569-F94429362922"

 **

    使用AFNetworking 、 SDWebImage 等等開源庫做網絡數據緩存的時候要注意可能會出現重復緩存(因為NSURLCache已經做了一次緩存,存在Disk中 !

 

    1. 使用 NSURLConnect,、UIWebView 。。 的都會使用導 NSURLCache 的緩存

 

    2. 當使用 NSURL 相關的框架,使用到緩存的時候,系統會在 Caches目錄下創建一個 跟 bundle identifer 同名的一個文件夾,以及里面的 Cache.db、Cache.db-shm、Cache.db-wal

    三個文件和文件夾 fsCacheData/

    ·fsCacheData/ 文件夾會在有需要緩存數據到文件的數據才會有,如緩存圖片

 

    3. 可以使用 [[NSURLCache sharedURLCache]removeAllCachedResponses] 清理這里所有的緩存數據。

*/

 

在AppDelegate.swift中添加

    func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
        
        /**
         * 設置網絡緩存
         **
        
         內存緩存 4M
         磁盤緩存 20M
         diskPath-》nil,會緩存到 cached 的 bundleId 目錄下
        
         SDWebImage 的緩存
        
         1. 緩存時間--1周
         2. 處理緩存文件,監聽系統退出到后台的事件
            - 遍歷緩存文件夾,刪除所有過期的文件
            - 繼續遍歷緩存文件夾,將最大的文件刪除,一直刪除到緩存文件的大小和指定的“磁盤限額”一致
        
         */
        
        let cache = NSURLCache(memoryCapacity: 4 * 1024 * 1024, diskCapacity: 20 * 1024 * 1024, diskPath: nil)
        NSURLCache .setSharedURLCache(cache)
        
        return true
    }


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM