Bagaimana cara mem-parse respons JSON dari Alamofire API di Swift?

125

Mengikuti kode yang saya tulis dan saya mendapatkan respons di JSON juga, tetapi jenis JSON adalah "AnyObject" dan saya tidak dapat mengubahnya menjadi Array sehingga saya dapat menggunakannya.

Alamofire.request(.POST, "MY URL", parameters:parameters, encoding: .JSON) .responseJSON
{
    (request, response, JSON, error) in

    println(JSON?)
}
Pengembang
sumber
Saya tidak mengecilkan pertanyaan Anda tetapi saya menganggapnya karena mengurai JSON adalah topik yang terlalu luas untuk memberikan jawaban yang jelas dan langsung. Coba pustaka ini bernama SwiftyJSON .
Isuru
@Isuru Tidak apa-apa! Saya telah melihat Perpustakaan itu tetapi saya menggunakan Alamofire! Tapi bisakah Anda mengirim saya kode sampel di mana Anda telah menggunakan SwiftyJson? Kode di sana tidak bekerja untuk saya!
Pengembang
Saya juga menggunakan SwiftyJSON bersama dengan Alamofire. Saya hanya memberikan tanggapan seperti ini let data = JSONValue(JSON!). Lalu saya bisa mengekstraksi nilai seperti ini data["Id"]. Dokumentasi SwiftyJSON memberikan contoh cara mengambil nilai-nilai tersebut dalam tipe yang diinginkan. Apa sebenarnya kesalahan yang Anda dapatkan?
Isuru

Jawaban:

160

Jawaban untuk Swift 2.0 Alamofire 3.0 seharusnya terlihat seperti ini:

Alamofire.request(.POST, url, parameters: parameters, encoding:.JSON).responseJSON
{ response in switch response.result {
                case .Success(let JSON):
                    print("Success with JSON: \(JSON)")

                    let response = JSON as! NSDictionary

                    //example if there is an id
                    let userId = response.objectForKey("id")!

                case .Failure(let error):
                    print("Request failed with error: \(error)")
                }
    }

https://github.com/Alamofire/Alamofire/blob/master/Documentation/Alamofire%203.0%20Migration%20Guide.md

UPDATE untuk Alamofire 4.0 dan Swift 3.0:

Alamofire.request(url, method: .post, parameters: parameters, encoding: JSONEncoding.default)
            .responseJSON { response in
                print(response)
//to get status code
                if let status = response.response?.statusCode {
                    switch(status){
                    case 201:
                        print("example success")
                    default:
                        print("error with response status: \(status)")
                    }
                }
//to get JSON return value
            if let result = response.result.value {
                let JSON = result as! NSDictionary
                print(JSON)
            }

        }
Joseph Geraghty
sumber
13
Bagaimana Anda mendapatkan konten JSON yang sebenarnya? Jenis objek apa ini? Desain dan dokumentasinya sangat tidak jelas sehingga saya tidak dapat menemukannya dan tidak dapat menemukan contoh di internet ...
Alex Worden
Saya menambahkan beberapa baris dalam jawaban saya yang akan membantu.
Joseph Geraghty
@JosephGeraghty memiliki parameter hasil pengkodean di kompiler mengatakan ada panggilan argumen tambahan ... Ada ide?
amariduran
@ Jch-duran tidak positif, tapi saya samar-samar ingat berlari ke sesuatu yang serupa beberapa waktu lalu. Saya pikir itu ada hubungannya dengan perpustakaan tidak diperbarui atau mungkin tidak saat ini dengan versi cepat. Memastikan Anda menggunakan versi terbaru mungkin membantu
Joseph Geraghty
1
@AlexWorden setuju, halaman ini membantu saya menjawab pertanyaan-pertanyaan itu dan memberikan solusi yang bagus: github.com/SwiftyJSON/SwiftyJSON
iljn
31

seperti disebutkan di atas Anda dapat menggunakan perpustakaan SwiftyJSON dan mendapatkan nilai-nilai Anda seperti yang saya lakukan di bawah ini

Alamofire.request(.POST, "MY URL", parameters:parameters, encoding: .JSON) .responseJSON
{
    (request, response, data, error) in

var json = JSON(data: data!)

       println(json)   
       println(json["productList"][1])                 

}

daftar produk json saya kembali dari skrip

{ "productList" :[

{"productName" : "PIZZA","id" : "1","productRate" : "120.00","productDescription" : "PIZZA AT 120Rs","productImage" : "uploads\/pizza.jpeg"},

{"productName" : "BURGER","id" : "2","productRate" : "100.00","productDescription" : "BURGER AT Rs 100","productImage" : "uploads/Burgers.jpg"}    
  ]
}

keluaran:

{
  "productName" : "BURGER",
  "id" : "2",
  "productRate" : "100.00",
  "productDescription" : "BURGER AT Rs 100",
  "productImage" : "uploads/Burgers.jpg"
}
Vikram Pote
sumber
Saya mencoba menggunakan SwiftyJson setelah menginstal tetapi memberikan 300 kesalahan pada file SwiftyJson, apakah ada yang menghadapi masalah? i ', menggunakan Xcode versi 6.2, ios versi 8.1, cocoaPods 36 sebagaimana disebutkan dalam dokumentasi [github] ( github.com/SwiftyJSON/SwiftyJSON ).
Sashi
2
Kawan Apa kesalahannya? Ajukan pertanyaan terpisah dan berikan beberapa detail. SwiftyJSON seindah sihir. Gunakan jika memungkinkan.
Zia
Benar-benar harus mengubah string json menjadi objek cepat beton sehingga Anda dapat menggunakannya dengan bersih secara alami. Mengakses bidang dengan nama string mereka konyol dan rentan terhadap kesalahan.
The Muffin Man
26

Swift 3, Alamofire 4.4, dan SwiftyJSON:

Alamofire.request(url, method: .get)
  .responseJSON { response in
      if response.data != nil {
        let json = JSON(data: response.data!)
        let name = json["people"][0]["name"].string
        if name != nil {
          print(name!)
        }
      }
  }

Itu akan mem-parsing input JSON ini:

{
  people: [
    { name: 'John' },
    { name: 'Dave' }
  ]
}
iljn
sumber
Ada juga plugin Alamofire Swifty-JSON-spesifik yang menghilangkan kebutuhan untuk JSON()konversi eksplisit : github.com/SwiftyJSON/Alamofire-SwiftyJSON
Robin Macharg
Ini membantu saya, tetapi saya memang memiliki beberapa masalah dengan metode JSON karena melempar Pengecualian
iGhost
24

Saya menemukan jawabannya di GitHub untuk Swift2

https://github.com/Alamofire/Alamofire/issues/641

Alamofire.request(.GET, URLString, parameters: ["foo": "bar"])
    .responseJSON { request, response, result in
        switch result {
        case .Success(let JSON):
            print("Success with JSON: \(JSON)")

        case .Failure(let data, let error):
            print("Request failed with error: \(error)")

            if let data = data {
                print("Response data: \(NSString(data: data, encoding: NSUTF8StringEncoding)!)")
            }
        }
    }
mick80234
sumber
3
Ini adalah versi yang benar untuk parsing Swift 2.0 + Alamofire JSON.
Saqib Omer
5
hmm saya masih gagal membangun pesan kesalahan: '(_, _, _) -> Void' tidak dapat dikonversi menjadi 'Response <AnyObject, NSError> -> Void'
alex
@alex Lihat jawaban ini untuk apa yang saya gunakan untuk menyelesaikannya.
Joseph
Terima kasih banyak ! Anda tidak tahu berapa banyak hal yang saya coba tampilkan dengan benar pesan respon dari server, penyelamat hidup!
thibaut noah
17

Saya bukan ahli JSON atau ahli Swift, tetapi berikut ini bekerja untuk saya. :) Saya telah mengekstrak kode dari aplikasi saya saat ini, dan hanya mengubah "MyLog ke println", dan menjorok dengan spasi untuk membuatnya ditampilkan sebagai blok kode (mudah-mudahan saya tidak merusaknya).

func getServerCourseVersion(){

    Alamofire.request(.GET,"\(PUBLIC_URL)/vtcver.php")
        .responseJSON { (_,_, JSON, _) in
          if let jsonResult = JSON as? Array<Dictionary<String,String>> {
            let courseName = jsonResult[0]["courseName"]
            let courseVersion = jsonResult[0]["courseVersion"]
            let courseZipFile = jsonResult[0]["courseZipFile"]

            println("JSON:    courseName: \(courseName)")
            println("JSON: courseVersion: \(courseVersion)")
            println("JSON: courseZipFile: \(courseZipFile)")

          }
      }
}

Semoga ini membantu.

Edit:

Untuk referensi, inilah yang dikembalikan Script PHP saya:

[{"courseName": "Training Title","courseVersion": "1.01","courseZipFile": "101/files.zip"}]
Ken
sumber
Ini harus menjadi jawaban yang dipilih walaupun Anda mungkin ingin memperbaruinya karena Alamofire telah memperbarui metode mereka sedikit
Snymax
10

cepat 3

pod 'Alamofire', '~> 4.4'
pod 'SwiftyJSON'

File json format:
{
    "codeAd": {
        "dateExpire": "2017/12/11",
        "codeRemoveAd":"1231243134"
        }
}

import Alamofire
import SwiftyJSON
    private func downloadJson() {
        Alamofire.request("https://yourlinkdownloadjson/abc").responseJSON { response in
            debugPrint(response)

            if let json = response.data {
                let data = JSON(data: json)
                print("data\(data["codeAd"]["dateExpire"])")
                print("data\(data["codeAd"]["codeRemoveAd"])")
            }
        }
    }
Giang
sumber
2

Saya menemukan cara untuk mengubah response.result.value (di dalam penutupan Alamofire responseJSON) ke dalam format JSON yang saya gunakan di aplikasi saya.

Saya menggunakan Alamofire 3 dan Swift 2.2.

Berikut kode yang saya gunakan:

    Alamofire.request(.POST, requestString,
                      parameters: parameters,
                      encoding: .JSON,
                      headers: headers).validate(statusCode: 200..<303)
                                       .validate(contentType: ["application/json"])
                                       .responseJSON { (response) in
        NSLog("response = \(response)")

        switch response.result {
        case .Success:
            guard let resultValue = response.result.value else {
                NSLog("Result value in response is nil")
                completionHandler(response: nil)
                return
            }

            let responseJSON = JSON(resultValue)

            // I do any processing this function needs to do with the JSON here

            // Here I call a completionHandler I wrote for the success case
        break
        case .Failure(let error):
            NSLog("Error result: \(error)")
            // Here I call a completionHandler I wrote for the failure case
            return
        }
Carl Smith
sumber
2

Saya biasanya menggunakan pustaka Gloss untuk membuat serial atau deserialize JSON di iOS. Misalnya, saya punya JSON yang terlihat seperti ini:

{"ABDC":[{"AB":"qwerty","CD":"uiop"}],[{"AB":"12334","CD":"asdf"}]}

Pertama, saya memodelkan array JSON di Gloss struct:

Struct Struct_Name: Decodable {
   let IJ: String?
   let KL: String?
   init?(json: JSON){
       self.IJ = "AB" <~~ json
       self.KL = "CD" <~~ json
   }
}

Dan kemudian di Alamofire responseJSON, saya melakukan hal berikut ini:

Alamofire.request(url, method: .get, paramters: parametersURL).validate(contentType: ["application/json"]).responseJSON{ response in
 switch response.result{
   case .success (let data):
    guard let value = data as? JSON,
       let eventsArrayJSON = value["ABDC"] as? [JSON]
    else { fatalError() }
    let struct_name = [Struct_Name].from(jsonArray: eventsArrayJSON)//the JSON deserialization is done here, after this line you can do anything with your JSON
    for i in 0 ..< Int((struct_name?.count)!) {
       print((struct_name?[i].IJ!)!)
       print((struct_name?[i].KL!)!)
    }
    break

   case .failure(let error):
    print("Error: \(error)")
    break
 }
}

Output dari kode di atas:

qwerty
uiop
1234
asdf
Luthfi Rahman
sumber
2

Cepat 5

class User: Decodable {

    var name: String
    var email: String
    var token: String

    enum CodingKeys: String, CodingKey {
        case name
        case email
        case token
    }

    public required init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: CodingKeys.self)
        self.name = try container.decode(String.self, forKey: .name)
        self.email = try container.decode(String.self, forKey: .email)
        self.token = try container.decode(String.self, forKey: .token)
    }
}

API Alamofire

    Alamofire.request("url.endpoint/path", method: .get, parameters: params, encoding: URLEncoding.queryString, headers: nil)
     .validate()
     .responseJSON { response in

        switch (response.result) {

            case .success( _):

            do {
                let users = try JSONDecoder().decode([User].self, from: response.data!)
                print(users)

            } catch let error as NSError {
                print("Failed to load: \(error.localizedDescription)")
            }

             case .failure(let error):
                print("Request error: \(error.localizedDescription)")
         }
Sazzad Hissain Khan
sumber
1

Ini dibangun dengan Xcode 10.1 dan Swift 4

Kombinasi sempurna "Alamofire" (4.8.1) dan "SwiftyJSON" (4.2.0). Pertama, Anda harus menginstal kedua pod

pod 'Alamofire' dan pod 'SwiftyJSON'

Respons server dalam format JSON:

{
  "args": {}, 
  "headers": {
    "Accept": "*/*", 
    "Accept-Encoding": "gzip;q=1.0, compress;q=0.5", 
    "Accept-Language": "en;q=1.0", 
    "Host": "httpbin.org", 
    "User-Agent": "AlamoFire TEST/1.0 (com.ighost.AlamoFire-TEST; build:1; iOS 12.1.0) Alamofire/4.8.1"
  }, 
  "origin": "200.55.140.181, 200.55.140.181", 
  "url": "https://httpbin.org/get"
}

Dalam hal ini saya ingin mencetak info "Host": "Host": "httpbin.org"

Alamofire.request("https://httpbin.org/get").validate().responseJSON { response in
        switch response.result {
        case .success:
            print("Validation Successful)")

            if let json = response.data {
                do{
                    let data = try JSON(data: json)
                    let str = data["headers"]["Host"]
                    print("DATA PARSED: \(str)")
                }
                catch{
                print("JSON Error")
                }

            }
        case .failure(let error):
            print(error)
        }
    }

Tetap Tenang dan Senang Kode 😎

iGhost
sumber
0

di swift 5 kita suka, Gunakan typealias untuk menyelesaikan. Typlealias tidak hanya digunakan untuk membersihkan kode.

typealias response = (Bool,Any?)->()


static func postCall(_ url : String, param : [String : Any],completion : @escaping response){
    Alamofire.request(url, method: .post, parameters: param, encoding: JSONEncoding.default, headers: [:]).responseJSON { (response) in

        switch response.result {
           case .success(let JSON):
               print("\n\n Success value and JSON: \(JSON)")

           case .failure(let error):
               print("\n\n Request failed with error: \(error)")

           }
    }
}
Talha Rasool
sumber
-10
 pod 'Alamofire'
 pod 'SwiftyJSON'
 pod 'ReachabilitySwift'



import UIKit
import Alamofire
import SwiftyJSON
import SystemConfiguration

class WebServiceHelper: NSObject {

    typealias SuccessHandler = (JSON) -> Void
    typealias FailureHandler = (Error) -> Void

    // MARK: - Internet Connectivity

    class func isConnectedToNetwork() -> Bool {

        var zeroAddress = sockaddr_in()
        zeroAddress.sin_len = UInt8(MemoryLayout<sockaddr_in>.size)
        zeroAddress.sin_family = sa_family_t(AF_INET)

        guard let defaultRouteReachability = withUnsafePointer(to: &zeroAddress, {
            $0.withMemoryRebound(to: sockaddr.self, capacity: 1) {
                SCNetworkReachabilityCreateWithAddress(nil, $0)
            }
        }) else {
            return false
        }

        var flags: SCNetworkReachabilityFlags = []
        if !SCNetworkReachabilityGetFlags(defaultRouteReachability, &flags) {
            return false
        }

        let isReachable = flags.contains(.reachable)
        let needsConnection = flags.contains(.connectionRequired)

        return (isReachable && !needsConnection)
    }

    // MARK: - Helper Methods

    class func getWebServiceCall(_ strURL : String, isShowLoader : Bool, success : @escaping SuccessHandler, failure : @escaping FailureHandler)
    {
        if isConnectedToNetwork() {

            print(strURL)

            if isShowLoader == true {

                AppDelegate.getDelegate().showLoader()
            }

            Alamofire.request(strURL).responseJSON { (resObj) -> Void in

                print(resObj)

                if resObj.result.isSuccess {
                    let resJson = JSON(resObj.result.value!)

                    if isShowLoader == true {
                        AppDelegate.getDelegate().dismissLoader()
                    }

                    debugPrint(resJson)
                    success(resJson)
                }
                if resObj.result.isFailure {
                    let error : Error = resObj.result.error!

                    if isShowLoader == true {
                        AppDelegate.getDelegate().dismissLoader()
                    }
                    debugPrint(error)
                    failure(error)
                }
            }
        }else {


            CommonMethods.showAlertWithError("", strMessage: Messages.NO_NETWORK, withTarget: (AppDelegate.getDelegate().window!.rootViewController)!)
        }
    }

    class func getWebServiceCall(_ strURL : String, params : [String : AnyObject]?, isShowLoader : Bool, success : @escaping SuccessHandler,  failure :@escaping FailureHandler){
        if isConnectedToNetwork() {

            if isShowLoader == true {
                AppDelegate.getDelegate().showLoader()
            }


            Alamofire.request(strURL, method: .get, parameters: params, encoding: JSONEncoding.default, headers: nil).responseJSON(completionHandler: {(resObj) -> Void in

                print(resObj)

                if resObj.result.isSuccess {
                    let resJson = JSON(resObj.result.value!)

                    if isShowLoader == true {
                        AppDelegate.getDelegate().dismissLoader()
                    }

                    success(resJson)
                }
                if resObj.result.isFailure {
                    let error : Error = resObj.result.error!

                    if isShowLoader == true {
                        AppDelegate.getDelegate().dismissLoader()
                    }

                    failure(error)
                }

            })
        }
    else {

            CommonMethods.showAlertWithError("", strMessage: Messages.NO_NETWORK, withTarget: (AppDelegate.getDelegate().window!.rootViewController)!)
    }

    }



    class func postWebServiceCall(_ strURL : String, params : [String : AnyObject]?, isShowLoader : Bool, success : @escaping SuccessHandler, failure :@escaping FailureHandler)
    {
        if isConnectedToNetwork()
        {

            if isShowLoader == true
            {
                AppDelegate.getDelegate().showLoader()
            }

            Alamofire.request(strURL, method: .post, parameters: params, encoding: JSONEncoding.default, headers: nil).responseJSON(completionHandler: {(resObj) -> Void in

                print(resObj)

                if resObj.result.isSuccess
                {
                    let resJson = JSON(resObj.result.value!)

                    if isShowLoader == true
                    {
                        AppDelegate.getDelegate().dismissLoader()
                    }

                    success(resJson)
                }

                if resObj.result.isFailure
                {
                    let error : Error = resObj.result.error!

                    if isShowLoader == true
                    {
                        AppDelegate.getDelegate().dismissLoader()
                    }

                    failure(error)
                }
            })
        }else {
            CommonMethods.showAlertWithError("", strMessage: Messages.NO_NETWORK, withTarget: (AppDelegate.getDelegate().window!.rootViewController)!)
        }
    }


    class func postWebServiceCallWithImage(_ strURL : String, image : UIImage!, strImageParam : String, params : [String : AnyObject]?, isShowLoader : Bool, success : @escaping SuccessHandler, failure : @escaping FailureHandler)
    {
        if isConnectedToNetwork() {
            if isShowLoader == true
            {
                AppDelegate.getDelegate().showLoader()
            }

            Alamofire.upload(
                multipartFormData: { multipartFormData in
                    if let imageData = UIImageJPEGRepresentation(image, 0.5) {
                        multipartFormData.append(imageData, withName: "Image.jpg")
                    }

                    for (key, value) in params! {

                        let data = value as! String

                        multipartFormData.append(data.data(using: String.Encoding.utf8)!, withName: key)
                        print(multipartFormData)
                    }
                },
                to: strURL,
                encodingCompletion: { encodingResult in
                    switch encodingResult {
                    case .success(let upload, _, _):
                        upload.responseJSON { response in
                            debugPrint(response)
                            //let datastring = String(data: response, encoding: String.Encoding.utf8)
                           // print(datastring)
                        }
                    case .failure(let encodingError):
                        print(encodingError)
                        if isShowLoader == true
                        {
                            AppDelegate.getDelegate().dismissLoader()
                        }

                        let error : NSError = encodingError as NSError
                        failure(error)
                    }

                    switch encodingResult {
                    case .success(let upload, _, _):
                        upload.responseJSON { (response) -> Void in

                            if response.result.isSuccess
                            {
                                let resJson = JSON(response.result.value!)

                                if isShowLoader == true
                                {
                                    AppDelegate.getDelegate().dismissLoader()
                                }

                                success(resJson)
                            }

                            if response.result.isFailure
                            {
                                let error : Error = response.result.error! as Error

                                if isShowLoader == true
                                {
                                    AppDelegate.getDelegate().dismissLoader()
                                }

                                failure(error)
                            }

                        }
                    case .failure(let encodingError):
                        if isShowLoader == true
                        {
                            AppDelegate.getDelegate().dismissLoader()
                        }

                        let error : NSError = encodingError as NSError
                        failure(error)
                    }
                }
            )
        }
        else
        {
            CommonMethods.showAlertWithError("", strMessage: Messages.NO_NETWORK, withTarget: (AppDelegate.getDelegate().window!.rootViewController)!)
        }
    }

}


==================================


Call Method


let aParams : [String : String] = [
                "ReqCode" : Constants.kRequestCodeLogin,
                ]

            WebServiceHelper.postWebServiceCall(Constants.BaseURL, params: aParams as [String : AnyObject]?, isShowLoader: true, success: { (responceObj) in


                if "\(responceObj["RespCode"])" != "1"
                {
                    let alert = UIAlertController(title: Constants.kAppName, message: "\(responceObj["RespMsg"])", preferredStyle: UIAlertControllerStyle.alert)
                    let OKAction = UIAlertAction(title: "OK", style: .default) { (action:UIAlertAction!) in
                    }
                    alert.addAction(OKAction)
                    self.present(alert, animated: true, completion: nil)
                }
                else
                {
                    let aParams : [String : String] = [
                        "Password" : self.dictAddLogin[AddLoginConstants.kPassword]!,
                        ]
                    CommonMethods.saveCustomObject(aParams as AnyObject?, key: Constants.kLoginData)

                }
                }, failure:
                { (error) in

                    CommonMethods.showAlertWithError(Constants.kALERT_TITLE_Error, strMessage: error.localizedDescription,withTarget: (AppDelegate.getDelegate().window!.rootViewController)!)
            })
        }
Nikunj Patel
sumber
6
Penjelasan tentang semua kode ini akan sangat membantu.
peter.swallow