介绍

SwiftyJSON是个使用Swift语言编写的开源库,可以让我们很方便地处理JSON数据(解析数据、生成数据)。
GitHub地址:https://github.com/SwiftyJSON/SwiftyJSON

优点

JSONSerializationSwiftyJSON 相比,在获取多层次结构的JSON数据时。SwiftyJSON不需要一直判断这个节点是否存在,是不是我们想要的class,下一个节点是否存在,是不是我们想要的class…。同时,SwiftyJSON内部会自动对optional(可选类型)进行拆包(Wrapping ),大大简化了代码。
下面通过几个样例作为演示。
(1)比如我们有一个如下的JSON数据,表示联系人集合:

  1. [{
  2. "name": "hangge",
  3. "age": 100,
  4. "phones": [{
  5. "name": "公司",
  6. "number": "123456"
  7. },
  8. {
  9. "name": "家庭",
  10. "number": "001"
  11. }
  12. ]
  13. },
  14. {
  15. "name": "big boss",
  16. "age": 1,
  17. "phones": [{
  18. "name": "公司",
  19. "number": "111111"
  20. }]
  21. }
  22. ]

为便于测试比较,我们先将JSON格式的字符串转为Data:

  1. let jsonStr = "[{\"name\": \"hangge\", \"age\": 100, \"phones\": [{\"name\": \"公司\",\"number\": \"123456\"}, {\"name\": \"家庭\",\"number\": \"001\"}]}, {\"name\": \"big boss\",\"age\": 1,\"phones\": [{ \"name\": \"公司\",\"number\": \"111111\"}]}]"
  2. if let jsonData = jsonStr.data(using: String.Encoding.utf8, allowLossyConversion: false) {
  3. //.........
  4. }


(2)使用JSONSerializationSwiftyJSON解析
比如我们要取第一条联系人的第一个电话号码,每个级别都判断就很麻烦,代码如下:

  1. if let userArray = try? JSONSerialization.jsonObject(with: jsonData, options: .allowFragments) as? [[String: AnyObject]],
  2. let phones = userArray?[0]["phones"] as? [[String: AnyObject]],
  3. let number = phones[0]["number"] as? String {
  4. // 找到电话号码
  5. print("第一个联系人的第一个电话号码:",number)
  6. }

即使使用optional来简化一下,代码也不少:

  1. if let userArray = try? JSONSerialization.jsonObject(with: jsonData,
  2. options: .allowFragments) as? [[String: AnyObject]],
  3. let number = (userArray?[0]["phones"] as? [[String: AnyObject]])?[0]["number"] as? String {
  4. // 找到电话号码
  5. print("第一个联系人的第一个电话号码:",number)
  6. }


(3)使用SwiftyJSON解析:
不用担心数组越界,不用判断节点,拆包什么的,代码如下:

  1. let json = try! JSON(data: jsonData)
  2. if let number = json[0]["phones"][0]["number"].string {
  3. // 找到电话号码
  4. print("第一个联系人的第一个电话号码:",number)
  5. }

如果没取到值,还可以走到错误处理来了,打印一下看看错在哪:

  1. let json = try! JSON(data: jsonData)
  2. if let number = json[0]["phones"][0]["number"].string {
  3. // 找到电话号码
  4. print("第一个联系人的第一个电话号码:",number)
  5. }else {
  6. // 打印错误信息
  7. print(json[0]["phones"][0]["number"])
  8. }

使用解析


除了解析本地的JSON数据,我们其实更常通过url地址获取远程数据并解析。
(1)与URLSession结合

  1. //创建URL对象
  2. let url = URL(string:"http://www.weather.com.cn/data/sk/101110101.html")
  3. //创建请求对象
  4. let request = URLRequest(url: url!)
  5. let dataTask = URLSession.shared.dataTask(with: request,
  6. completionHandler: {(data, response, error) -> Void in
  7. if error != nil{
  8. print(error)
  9. }else{
  10. let json = try! JSON(data: data!)
  11. if let number = json[0]["phones"][0]["number"].string {
  12. // 找到电话号码
  13. print("第一个联系人的第一个电话号码:",number)
  14. }
  15. }
  16. }) as URLSessionTask
  17. //使用resume方法启动任务
  18. dataTask.resume()


(2)与Alamofire结合

  1. //创建URL对象
  2. let url = URL(string:"http://www.hangge.com/getJsonData.php")!
  3. Alamofire.request(url).validate().responseJSON { response in
  4. switch response.result.isSuccess {
  5. case true:
  6. if let value = response.result.value {
  7. let json = JSON(value)
  8. if let number = json[0]["phones"][0]["number"].string {
  9. // 找到电话号码
  10. print("第一个联系人的第一个电话号码:",number)
  11. }
  12. }
  13. case false:
  14. print(response.result.error)
  15. }
  16. }

初始化

  • 直接通过数据:

    1. let json = JSON(data: jsonData)
  • 通过object: Any初始化

这里需要注意的是参数object必须包含如下属性:NSString/String, NSNumber/Int/Float/Double/Bool, NSArray/Array, NSDictionary/Dictionary, or NSNull;而且这里的JsonString是框架是不会将它转换的你应该通过init(parseJson: String)方式处理,并且所有的dictionary 所包含的keys最好是NSString/String

  1. let json = JSON(jsonObject as Any)
  • 通过JsonString 初始化
    1. let json =JSON(parseJSON: jsonString)

    获取值

    1. 可选值获取(Optional getter)

    通过.number、.string、.bool、.int、.uInt、.float、.double、.array、.dictionary、int8、Uint8、int16、Uint16、int32、Uint32、int64、Uint64等方法获取到的是可选择值,我们需要自行判断是否存在,同时不存在的话可以获取具体的错误信息。 ```swift //int if let age = json[0][ “age” ].int { print (age) } else { //打印错误信息 print (json[0][ “age” ]) }

//String if let name = json[0][ “name” ].string { print (name) } else { //打印错误信息 print (json[0][ “name” ]) }

  1. <a name="U6o61"></a>
  2. #### 2.不可选值获取(Non-optional getter)
  3. 使用 **xxxValue** 这样的属性获取值,如果没获取到的话会返回一个默认值。省得我们再判断拆包了。
  4. ```swift
  5. //If not a Number or nil, return 0
  6. let age: Int = json[0][ "age" ].intValue
  7. //If not a String or nil, return ""
  8. let name: String = json[0][ "name" ].stringValue
  9. //If not a Array or nil, return []
  10. let list: Array < JSON > = json[0][ "phones" ].arrayValue
  11. //If not a Dictionary or nil, return [:]
  12. let phone: Dictionary < String , JSON > = json[0][ "phones" ][0].dictionaryValue

3. 获取原始数据(Raw object)

  1. let jsonObject = json.object as AnyObject
  2. let jsonObject = json.rawValue as AnyObject
  3. //JSON转化为Data
  4. let data = json.rawData()
  5. //JSON转化为String字符串
  6. if let string = json.rawString() {
  7. //Do something you want
  8. }
  9. //JSON转化为Dictionary字典([String: AnyObject]?)
  10. if let dic = json.dictionaryObject {
  11. //Do something you want
  12. }
  13. //JSON转化为Array数组([AnyObject]?)
  14. if let arr = json.arrayObject {
  15. //Do something you want
  16. }

设置值

  1. json[0][ "age" ].int = 101
  2. json[0][ "name" ].string = "hangge.com"
  3. json[0][ "phones" ].arrayObject = [[ "name" : "固话" , "number" :110],[ "name" : "手机" , "number" :120]]
  4. json[0][ "phones" ][0].dictionaryObject = [ "name" : "固话" , "number" :100]

下标访问(Subscript)

可以通过数字、字符串、数组类型的下标访问JSON数据的层级与属性。比如下面三种方式的结果都是一样的:

  1. //方式1
  2. let number = json[0]["phones"][0]["number"].stringValue
  3. //方式2
  4. let number = json[0,"phones",0,"number"].stringValue
  5. //方式3
  6. let keys:[JSONSubscriptType] = [0,"phones",0,"number"]
  7. let number = json[keys].stringValue

循环遍历JSON对象中的所有数据

  1. //(1)如果JSON数据是数组类型(Array)
  2. for (index,subJson):(String, JSON) in json {
  3. print("\(index):\(subJson)")
  4. }
  5. //(2)如果JSON数据是字典类型(Dictionary)
  6. for (key,subJson):(String, JSON) in json[0] {
  7. print("\(key):\(subJson)")
  8. }

构造创建JSON对象数据

  1. // (1)空的JSON对象
  2. let json: JSON = nil
  3. // (2)使用简单的数据类型创建JSON对象
  4. //StringLiteralConvertible
  5. let json: JSON = "I'm a son"
  6. //IntegerLiteralConvertible
  7. let json: JSON = 12345
  8. //BooleanLiteralConvertible
  9. let json: JSON = true
  10. //FloatLiteralConvertible
  11. let json: JSON = 2.8765
  12. //(3)使用数组或字典数据创建JSON对象
  13. //DictionaryLiteralConvertible
  14. let json: JSON = ["I":"am", "a":"son"]
  15. //ArrayLiteralConvertible
  16. let json: JSON = ["I", "am", "a", "son"]
  17. //Array & Dictionary
  18. var json: JSON = ["name": "Jack", "age": 25, "list": ["a", "b", "c", ["what": "this"]]]
  19. json["list"][3]["what"] = "that"
  20. json["list",3,"what"] = "that"
  21. let path:[JSONSubscriptType] = ["list",3,"what"]
  22. json[path] = "that"

其他方法

exists

  1. // 判断是否存在
  2. if json["name"].exists()

merge

  1. let original: JSON = [
  2. "first_name": "Theo",
  3. "age": 20,
  4. "skills": ["Coding", "Reading"],
  5. "address": [
  6. "street": "Software St",
  7. "zip": "210046",
  8. ]
  9. ]
  10. let update: JSON = [
  11. "last_name": "Tsao",
  12. "age": 22,
  13. "skills": ["Writing"],
  14. "address": [
  15. "zip": "210012",
  16. "city": "Nanjing"
  17. ]
  18. ]
  19. let updated = original.merge(with: update)

输出:

  1. [
  2. "first_name": "Theo",
  3. "last_name": "Tsao",
  4. "age": 22,
  5. "skills": ["Coding", "Reading", "Writing"],
  6. "address": [
  7. "street": "Software St",
  8. "zip": "210012",
  9. "city": "Nanjing"
  10. ]
  11. ]