빙수왕의 개발일지

타입 나눠서 decoding하기 본문

개발/iOS

타입 나눠서 decoding하기

빙수킹 2023. 1. 14. 19:39

서버에서 내려주는 커피 리스트가 있다. 이 커피 리스트를 디코딩한다고 가정하자.

커피 종류는 2가지가 있고, 이 2가지는 내부 프로퍼티 값에 따라 구분된다. (또는 각 다른 프로퍼티를 가진다)

ex) [StructA(), StructA(), StructB(), StructA(), StructB(), ... ] 

 

서버에서 다 같은 형태로 내려주는 아이를 앱에서는 구분하고 싶을 때!!

각 타입을 다른 struct로 디코딩 하고싶다면 어떻게 할까?

 

CoffeeDecoder struct를 따로 만들어준다.

 

예시)

Coffee 의 ingredients 값에 따라 형태 나눌 수 있다.

아래 예시에서는 두가지 모두 Coffee struct 형태를 사용하지만, 각 Coffee 종류의 보유 프로퍼티가 다르다면 Struct 2가지를 나눠서 분리도 가능하다.

enum CoffeeDecoder: Decodable {

    case basedOnCoffee(Coffee)
    case others(Coffee)

    init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: Coffee.CodingKeys.self)

        // ingredients에 "Coffee"가 있는 부류, 아닌 부류 나누기
        let ingredients = try container.decodeIfPresent([String].self, forKey: Coffee.CodingKeys.ingredients) ?? []

        let basedOnCoffee = ingredients.contains {
            $0 == "Coffee"
        }

        let coffee = try Coffee(from: decoder)
        if basedOnCoffee {
            self = .basedOnCoffee(coffee)
        } else {
            self = .others(coffee)
        }
    }

    var coffee: Coffee {
        switch self {
        case .basedOnCoffee(let basedOnCoffee):
            return basedOnCoffee
        case .others(let others):
            return others
        }
    }
}

 

Coffee Struct

struct Coffee: Decodable {

    let title: String?
    let description: String
    let ingredients: [String]
    let image: String
    let id: Int
}

 

호출하는 부분

let jsonData = try Data(contentsOf: location)
let decoded = try JSONDecoder().decode([CoffeeDecoder].self, from: jsonData)