web-dev-qa-db-fra.com

Swift peut-il convertir une donnée de classe/structure en dictionnaire?

Par exemple:

class Test {
    var name: String;
    var age: Int;
    var height: Double;
    func convertToDict() -> [String: AnyObject] { ..... }
}

let test = Test();
test.name = "Alex";
test.age = 30;
test.height = 170;

let dict = test.convertToDict();

dict aura le contenu:

{"name": "Alex", "age": 30, height: 170}

Est-ce possible à Swift?

Et puis-je accéder à une classe comme un dictionnaire, par exemple en utilisant probablement:

test.value(forKey: "name");

Ou quelque chose comme ça?

Merci.

7
Chen Li Yong

Vous pouvez simplement ajouter une propriété calculée à votre struct pour renvoyer une Dictionary avec vos valeurs. Notez que le type de dictionnaire natif Swift n’a aucune méthode appelée value(forKey:). Vous devez convertir votre Dictionary en NSDictionary:

struct Test {
    let name: String
    let age: Int
    let height: Double
    var dictionary: [String: Any] {
        return ["name": name,
                "age": age,
                "height": height]
    }
    var nsDictionary: NSDictionary {
        return dictionary as NSDictionary
    }
}

Vous pouvez également étendre le protocole Encodable comme suggéré dans la réponse liée envoyée par @ColGraff pour le rendre universel à toutes les structures Encodable:

struct JSON {
    static let encoder = JSONEncoder()
}
extension Encodable {
    subscript(key: String) -> Any? {
        return dictionary[key]
    }
    var dictionary: [String: Any] {
        return (try? JSONSerialization.jsonObject(with: JSON.encoder.encode(self))) as? [String: Any] ?? [:]
    }
}

struct Test: Codable {
    let name: String
    let age: Int
    let height: Double
}

let test = Test(name: "Alex", age: 30, height: 170)
test["name"]    // Alex
test["age"]     // 30
test["height"]  // 170
24
Leo Dabus

Vous pouvez utiliser Reflection and Mirror comme ceci pour le rendre plus dynamique et vous assurer de ne pas oublier une propriété.

struct Person {
  var name:String
  var position:Int
  var good : Bool
  var car : String

  var asDictionary : [String:Any] {
    let mirror = Mirror(reflecting: self)
    let dict = Dictionary(uniqueKeysWithValues: mirror.children.lazy.map({ (label:String?,value:Any) -> (String,Any)? in
      guard label != nil else { return nil }
      return (label!,value)
    }).compactMap{ $0 })
    return dict
  }
}


let p1 = Person(name: "Ryan", position: 2, good : true, car:"Ford")
print(p1.asDictionary)

["nom": "Ryan", "position": 2, "bon": vrai, "voiture": "Ford"]

0
Ryan Heitner