Преобразовать словарь в JSON в Swift

129

Я создал следующий словарь:

var postJSON = [ids[0]:answersArray[0], ids[1]:answersArray[1], ids[2]:answersArray[2]] as Dictionary

и я получаю:

[2: B, 1: A, 3: C]

Итак, как я могу преобразовать его в JSON?

Теги:
serialization

10 ответов

185

Swift 3.0

С Swift 3 имя NSJSONSerialization и его методы изменились в соответствии с Руководством по дизайну Swift API.

let dic = ["2": "B", "1": "A", "3": "C"]

do {
    let jsonData = try JSONSerialization.data(withJSONObject: dic, options: .prettyPrinted)
    // here "jsonData" is the dictionary encoded in JSON data

    let decoded = try JSONSerialization.jsonObject(with: jsonData, options: [])
    // here "decoded" is of type `Any`, decoded from JSON data

    // you can now cast it with the right type        
    if let dictFromJSON = decoded as? [String:String] {
        // use dictFromJSON
    }
} catch {
    print(error.localizedDescription)
}

Swift 2.x

do {
    let jsonData = try NSJSONSerialization.dataWithJSONObject(dic, options: NSJSONWritingOptions.PrettyPrinted)
    // here "jsonData" is the dictionary encoded in JSON data

    let decoded = try NSJSONSerialization.JSONObjectWithData(jsonData, options: [])
    // here "decoded" is of type `AnyObject`, decoded from JSON data

    // you can now cast it with the right type 
    if let dictFromJSON = decoded as? [String:String] {
        // use dictFromJSON
    }
} catch let error as NSError {
    print(error)
}

Swift 1

var error: NSError?
if let jsonData = NSJSONSerialization.dataWithJSONObject(dic, options: NSJSONWritingOptions.PrettyPrinted, error: &error) {
    if error != nil {
        println(error)
    } else {
        // here "jsonData" is the dictionary encoded in JSON data
    }
}

if let decoded = NSJSONSerialization.JSONObjectWithData(jsonData, options: nil, error: &error) as? [String:String] {
    if error != nil {
        println(error)
    } else {
        // here "decoded" is the dictionary decoded from JSON data
    }
}

  • 0
    Я получаю следующее [2: A, 1: A, 3: A] . Но как насчет фигурных скобок?
  • 1
    Я не понимаю ваш вопрос. Какие фигурные скобки? Вы спрашивали о кодировании словаря в JSON, и это мой ответ.
Показать ещё 4 комментария
122

Вы делаете неправильное предположение. То, что отладчик/игровая площадка отображает ваш словарь в квадратных скобках (именно так Cocoa отображает словари), не означает, что формат JSON выводится именно так.

Вот пример кода, который преобразует словарь строк в JSON:

Версия Swift 3:

import Foundation

let dictionary = ["aKey": "aValue", "anotherKey": "anotherValue"]
if let theJSONData = try? JSONSerialization.data(
    withJSONObject: dictionary,
    options: []) {
    let theJSONText = String(data: theJSONData,
                               encoding: .ascii)
    print("JSON string = \(theJSONText!)")
}

Чтобы отобразить вышеперечисленное в "довольно печатном" формате, измените строку параметров на:

    options: [.prettyPrinted]

Или в синтаксисе Swift 2:

import Foundation

let dictionary = ["aKey": "aValue", "anotherKey": "anotherValue"]
let theJSONData = NSJSONSerialization.dataWithJSONObject(
  dictionary ,
  options: NSJSONWritingOptions(0),
  error: nil)
let theJSONText = NSString(data: theJSONData!,
  encoding: NSASCIIStringEncoding)
println("JSON string = \(theJSONText!)")

Выход этого

"JSON string = {"anotherKey":"anotherValue","aKey":"aValue"}"

Или в симпатичном формате:

{
  "anotherKey" : "anotherValue",
  "aKey" : "aValue"
}

Словарь заключен в фигурные скобки в выводе JSON, как и следовало ожидать.

РЕДАКТИРОВАТЬ:

В синтаксисе Swift 3/4 приведенный выше код выглядит следующим образом:

  let dictionary = ["aKey": "aValue", "anotherKey": "anotherValue"]
    if let theJSONData = try?  JSONSerialization.data(
      withJSONObject: dictionary,
      options: .prettyPrinted
      ),
      let theJSONText = String(data: theJSONData,
                               encoding: String.Encoding.ascii) {
          print("JSON string = \n\(theJSONText)")
    }
  }
  • 0
    Обычная строка Swift также работает в объявлении JSONText.
  • 0
    @thefredelement, как конвертировать NSData напрямую в строку Swift? Преобразование данных в строку является функцией NSString.
Показать ещё 11 комментариев
23

Мой ответ на ваш вопрос ниже

let dict = ["0": "ArrayObjectOne", "1": "ArrayObjecttwo", "2": "ArrayObjectThree"]

var error : NSError?

let jsonData = try! NSJSONSerialization.dataWithJSONObject(dict, options: NSJSONWritingOptions.PrettyPrinted)

let jsonString = NSString(data: jsonData, encoding: NSUTF8StringEncoding)! as String

print(jsonString)

Ответ

{
  "0" : "ArrayObjectOne",
  "1" : "ArrayObjecttwo",
  "2" : "ArrayObjectThree"
}
17

Иногда необходимо распечатать ответ сервера для целей отладки. Здесь функция, которую я использую:

extension Dictionary {

    var json: String {
        let invalidJson = "Not a valid JSON"
        do {
            let jsonData = try JSONSerialization.data(withJSONObject: self, options: .prettyPrinted)
            return String(bytes: jsonData, encoding: String.Encoding.utf8) ?? invalidJson
        } catch {
            return invalidJson
        }
    }

    func printJson() {
        print(json)
    }

}

Пример использования:

(lldb) po dictionary.printJson()
{
  "InviteId" : 2,
  "EventId" : 13591,
  "Messages" : [
    {
      "SenderUserId" : 9514,
      "MessageText" : "test",
      "RecipientUserId" : 9470
    },
    {
      "SenderUserId" : 9514,
      "MessageText" : "test",
      "RecipientUserId" : 9470
    }
  ],
  "TargetUserId" : 9470,
  "InvitedUsers" : [
    9470
  ],
  "InvitingUserId" : 9514,
  "WillGo" : true,
  "DateCreated" : "2016-08-24 14:01:08 +00:00"
}
15

Свифт 4:

let dic = ["2": "B", "1": "A", "3": "C"]
let encoder = JSONEncoder()
if let jsonData = try? encoder.encode(dic) {
    if let jsonString = String(data: jsonData, encoding: .utf8) {
        print(jsonString)
    }
}

Обратите внимание, что ключи и значения должны реализовывать Codable. Строки, целые и двойные числа (и более) уже можно Codable. См. Кодирование и декодирование пользовательских типов.

7

Swift 3:

let jsonData = try? JSONSerialization.data(withJSONObject: dict, options: [])
let jsonString = String(data: jsonData!, encoding: .utf8)!
print(jsonString)
  • 0
    Это приведет к сбою, если какая-либо часть будет нулевой, очень плохая практика для принудительного развертывания результатов. // В любом случае такая же информация (без сбоя) уже есть в других ответах, пожалуйста, избегайте публикации дублированного контента. Благодарю.
4

Ответ на ваш вопрос ниже:

Swift 2.1

     do {
          if let postData : NSData = try NSJSONSerialization.dataWithJSONObject(dictDataToBeConverted, options: NSJSONWritingOptions.PrettyPrinted){

          let json = NSString(data: postData, encoding: NSUTF8StringEncoding)! as String
          print(json)}

        }
        catch {
           print(error)
        }
1

Вот простое расширение, чтобы сделать это:

https://gist.github.com/stevenojo/0cb8afcba721838b8dcb115b846727c3

extension Dictionary {
    func jsonString() -> NSString? {
        let jsonData = try? JSONSerialization.data(withJSONObject: self, options: [])
        guard jsonData != nil else {return nil}
        let jsonString = String(data: jsonData!, encoding: .utf8)
        guard jsonString != nil else {return nil}
        return jsonString! as NSString
    }

}
  • 0
    Этот ответ на самом деле не добавляет ничего, чего нет в другом ответе, кроме помещения сериализации JSON в расширение. Если бы у меня был выбор, я бы предпочел использовать JSONEncoder вместо библиотеки JSONSerialization. Использование охранников также кажется более неуклюжим, чем использование «если позволено».
0

Свифт 4 Расширение Dictionary.

extension Dictionary {
    var jsonStringRepresentaiton: String? {
        guard let theJSONData = try? JSONSerialization.data(withJSONObject: self,
                                                            options: [.prettyPrinted]) else {
            return nil
        }

        return String(data: theJSONData, encoding: .ascii)
    }
}
  • 0
    Это хороший и многократно используемый способ решения проблемы, но небольшое объяснение поможет новичкам лучше понять его.
0
private func convertDictToJson(dict : NSDictionary) -> NSDictionary?
{
    var jsonDict : NSDictionary!

    do {
        let jsonData = try JSONSerialization.data(withJSONObject:dict, options:[])
        let jsonDataString = String(data: jsonData, encoding: String.Encoding.utf8)!
        print("Post Request Params : \(jsonDataString)")
        jsonDict = [ParameterKey : jsonDataString]
        return jsonDict
    } catch {
        print("JSON serialization failed:  \(error)")
        jsonDict = nil
    }
    return jsonDict
}
  • 1
    Несколько ошибок здесь. Зачем использовать NSDictionary Фонда вместо словаря Swift? Кроме того, зачем возвращать новый словарь со строкой в качестве значения, а не возвращать фактические данные JSON? Это не имеет смысла. Также неявно развернутый необязательный параметр, возвращаемый как необязательный, на самом деле не очень хорошая идея.

Ещё вопросы

Сообщество Overcoder
Наверх
Меню