Как разобрать JSON с помощью swift 4 - PullRequest
0 голосов
/ 16 мая 2018

Я путаюсь с детализацией фруктов

{
  "fruits": [
    {
      "id": "1",
      "image": "https://cdn1.medicalnewstoday.com/content/images/headlines/271/271157/bananas.jpg",
      "name": "Banana"
    },
    {
      "id": "2",
      "image": "http://soappotions.com/wp-content/uploads/2017/10/orange.jpg",
      "title": "Orange"
    }
  ]
}

Хотите проанализировать JSON с помощью "Decodable"

struct Fruits: Decodable {
    let Fruits: [fruit]
}
struct fruit: Decodable {
    let id: Int?
    let image: String?
    let name: String?
}

let url = URL(string: "https://www.JSONData.com/fruits")
        URLSession.shared.dataTask(with: url!) { (data, response, error) in

            guard let data = data else { return }

            do{
                let fruits = try JSONDecoder().decode(Fruits.self, from: data)
                print(Fruits)

            }catch {
                print("Parse Error")
            }

Также вы можете предложить мне библиотеку cocoapod для быстрой загрузки изображений

Ответы [ 2 ]

0 голосов
/ 01 октября 2018
        Model sample:
        public struct JsonData: Codable{
            let data: [Data]?
            let meta: MetaValue?
            let linksData: LinksValue?

            private enum CodingKeys: String, CodingKey{
                case data
                case meta
                case linksData = "links"
            }
        }


     enum BackendError: Error {
    case urlError(reason: String)
    case objectSerialization(reason: String)
}

struct APIServiceRequest {

    static func serviceRequest<T>(reqURLString: String,
                                  resultStruct: T.Type,
                                  completionHandler:@escaping ((Any?, Error?) -> ())) where T : Decodable {
        guard let url = URL(string: reqURLString) else {
            print("Error: cannot create URL")
            let error = BackendError.urlError(reason: "Could not construct URL")
            completionHandler(nil, error)
            return
        }
        let urlRequest = URLRequest(url: url)

        let session = URLSession.shared

        let task = session.dataTask(with: urlRequest) { (data, response, error) in
            guard error == nil else {
                completionHandler(nil, error)
                return
            }

            guard let responseData = data else {
                print("Error: did not receive data")
                let error = BackendError.objectSerialization(reason: "No data in response")
                completionHandler(nil, error)
                return
            }

            let decoder = JSONDecoder()
            do {
                let books = try decoder.decode(resultStruct, from: responseData)
                completionHandler(books, nil)
            } catch {
                print("error trying to convert data to JSON")
                print(error)
                completionHandler(nil, error)
            }
        }

        task.resume()
    }

}

Метод POST

func loginWS(endpoint: String, completionHandler: @escaping (Any?) -> Swift.Void) {

    guard let sourceUrl = URL(string: endpoint) else { return }
    let request = NSMutableURLRequest(url: sourceUrl)

    let session = URLSession.shared
    request.httpMethod = "POST"
    request.addValue(vehiceHeader, forHTTPHeaderField: "X-Vehicle-Type")
    request.addValue(contentHeader, forHTTPHeaderField: "Content-Type")

    let task = session.dataTask(with: request as URLRequest) { data, response, error in

        guard let data = data else { return }
        do {
            let responseData = try JSONDecoder().decode(JsonData.self, from: data)
            print("response data:", responseData)
            completionHandler(responseData)
        } catch let err {
            print("Err", err)
        }
        }.resume()
}
0 голосов
/ 17 мая 2018

Проблема, с которой вы сталкиваетесь, заключается в том, что ваш JSON возвращает разные данные для ваших фруктов.

Для 1-го идентификатора возвращается String с именем name, но во 2-м возвращается строковое значение title.

Кроме того, при анализе JSON идентификатор выглядит как String, а не Int.

Таким образом, у вас есть два необязательных значения из ваших данных.

Таким образом, ваша Декодируемая Структура должна выглядеть примерно так:

struct Response: Decodable {
    let fruits: [Fruits]

}

struct Fruits: Decodable {
    let id: String
    let image: String
    let name: String?
    let title: String?
}

Поскольку ваш URL-адрес недействителен, я создал файл JSON в своем основном комплекте и смог его правильно проанализировать, например:

/// Parses The JSON
func parseJSON(){

    if let path = Bundle.main.path(forResource: "fruits", ofType: "json") {

        do {
            let data = try Data(contentsOf: URL(fileURLWithPath: path), options: .mappedIfSafe)
            let jsonResult = try JSONDecoder().decode(Response.self, from: data)

            let fruitsArray = jsonResult.fruits

            for fruit in fruitsArray{

                print("""
                    ID = \(fruit.id)
                    Image = \(fruit.image)
                    """)

                if let validName = fruit.name{
                     print("Name = \(validName)")
                }

                if let validTitle = fruit.title{
                    print("Title = \(validTitle)")
                }


            }

        } catch {
           print(error)
        }
    }
}

Надеюсь, это поможет ...

...