现在很多iOS项目的开发开始转向Swift语言。 相信 Swift语言很快会成为iOS工程师 必备技能。 字典转模型, 模型转转字典在开发过程中扮演非常重要的角色。 今天就和大家分享一下使用Swift,如何进行字典模型互转。 为了让工作做到极致,这里先提供一个工具 JSONExport。该工具能够使用json数据生成对应的模型类文件,支持oc和Swift,非常不错。

功能:

1,字典-->模型 :最简单的形式

class User: NSObject {  //模型类
var name:String?
var icon:String? // print时会调用。相当于java中的 toString()。为了代码整洁下面的模型去了这个计算属性。
override internal var description: String {
return "name: \(name) \n icon:\(icon) \n"
}
} func func1(){
let dict = ["name":"Jack","icon":"lufy.png"]
if let user = User.objectWithKeyValues(dict) as? User{
print("\(user)")
}
}
输出: name: Optional("Jack")
icon: Optional("lufy.png")

2,字典-->模型 :模型中包裹模型

//模型类
class Status :NSObject {
var text:String?
var user:User? //与 1 中的模型相同
var retweetedStatus:Status?
} func func2(){
let dict = ["text":"Agree!Nice weather!",
"user":["name":"Jack","icon":"lufy.png"],
"retweetedStatus":["text":"Nice weather!",
"user":["name":"Rose","icon":"nami.png"]]
] if let status = Status.objectWithKeyValues(dict) as? Status{
print("\(status)")
}
}
输出:
text:Optional("Agree!Nice weather!")
user:Optional(name: Optional("Jack") icon:Optional("lufy.png"))
retweetedStatus:Optional(text:Optional("Nice weather!")
user:Optional(name: Optional("Rose")icon:Optional("nami.png"))
retweetedStatus:nil)

3,字典-->模型: 字典中包裹数组, 数组中的元素是 一个模型对应的字典

//模型类, 必须遵守DictModelProtocol协议, 并实现customClassMapping方法。
class UserGroup: NSObject,DictModelProtocol {
var groupName:String?; //团队名称
var numbers:NSArray? //成员,保存User实例
static func customClassMapping() -> [String: String]?{
return ["numbers":"User"]; //指定numbers数组中的元素类型是User
}
} func func3(){
let dict = ["groupName":"Dream Team",
"numbers":[["name":"Jack","icon":"lufy.png"],
["name":"Rose","icon":"nami.png"]]
]
if let group = UserGroup.objectWithKeyValues(dict){
print("\(group)")
}
} 输出: groupName:Optional("Dream Team")
numbers:Optional((
"name: Optional(\"Jack\") \n icon:Optional(\"lufy.png\") \n",
"name: Optional(\"Rose\") \n icon:Optional(\"nami.png\") \n"
))

4,字典-->模型: 将一个字典数组转成模型数组

func func4(){
let arrayOfStatus = [["text":"Agree!Nice weather!",
"user":["name":"Jack",
"icon":"lufy.png"
],
"retweetedStatus":["text":"Nice weather!",
"user":["name":"Rose",
"icon":"nami.png"
]
]
],
["text":"2___Agree!Nice weather!",
"user":["name":"2___Jack",
"icon":"2___lufy.png"
],
"retweetedStatus":["text":"2___Nice weather!",
"user":["name":"2___Rose",
"icon":"2___nami.png"
]
]
]] if let status = Status.objectArrayWithKeyValuesArray(arrayOfStatus){
for item in status{ //打印出数组的元素
print(item)
}
}
}
输出:
text:Optional("Agree!Nice weather!")
user:Optional(name: Optional("Jack")icon:Optional("lufy.png"))
retweetedStatus:Optional(text:Optional("Nice weather!")
user:Optional(name: Optional("Rose") icon:Optional("nami.png"))
retweetedStatus:nil
) text:Optional("2___Agree!Nice weather!")
user:Optional(name: Optional("2___Jack")icon:Optional("2___lufy.png"))
retweetedStatus:Optional(text:Optional("2___Nice weather!")
user:Optional(name: Optional("2___Rose")icon:Optional("2___nami.png"))
retweetedStatus:nil
)

5 模型-->字典: 最简单形式

func func5(){
let user = User()
user.name = "hejunm"
user.icon = "my.png"
if let dict = user.keyValues{
do{ //转化为JSON 字符串,打印出来更直观
let data = try NSJSONSerialization.dataWithJSONObject(dict, options: .PrettyPrinted)
print(NSString(data: data, encoding: NSUTF8StringEncoding))
}catch{}
}
}
输出:
Optional({
"icon" : "my.png",
"name" : "hejunm"
})

6 模型-->字典: 模型中还有模型

func func6(){
let user = User()
user.name = "retweeted user hejunm"
user.icon = "my.png" let retweetedStatus = Status(); //转发微博
retweetedStatus.text = "this is retweeted status";
retweetedStatus.user = user let oriUser = User()
oriUser.name = "original user"
oriUser.icon = "my.png" let oriStatus = Status(); //原微博
oriStatus.text = "this is original status"
oriStatus.user = oriUser
oriStatus.retweetedStatus = retweetedStatus let dict = oriStatus.keyValues
do{ //转化为JSON 字符串
var data = try NSJSONSerialization.dataWithJSONObject(dict!, options: .PrettyPrinted)
print(NSString(data: data, encoding: NSUTF8StringEncoding)) }catch{ }
} 输出:
Optional({
"text" : "this is original status",
"user" : {
"icon" : "my.png",
"name" : "original user"
},
"retweetedStatus" : {
"text" : "this is retweeted status",
"user" : {
"icon" : "my.png",
"name" : "retweeted user hejunm"
}
}
})

7,模型-->字典 : 模型数组转字典数组

func func7(){

        let user1 = User()
user1.name = "hejunm_1"
user1.icon = "my.png_1" let user2 = User()
user2.name = "hejunm_2"
user2.icon = "my.png_2" let userArray = [user1,user2] as NSArray if let dicts = userArray.keyValuesArray{
do{
let data = try NSJSONSerialization.dataWithJSONObject(dicts, options: .PrettyPrinted) //转成json字符串
print(NSString(data: data, encoding: NSUTF8StringEncoding))
}catch{ }
}
} 输出:
Optional([
{
"icon" : "my.png_1",
"name" : "hejunm_1"
},
{
"icon" : "my.png_2",
"name" : "hejunm_2"
}
])

源码

字典-->模型

//
// HE_Dict2Model.swift
// HEExtention
//
// Created by 贺俊孟 on 16/4/27.
// Copyright © 2016年 贺俊孟. All rights reserved.
// 字典传模型 import Foundation /** 当字典中存在数组, 并且数组中保存的值得类型是字典, 那么就需要指定数组中的字典对应的类类型。
这里以键值对的形式保存
eg 字典如下:
key: [[key1:value1, key2:value2],[key1:value3, key2:value4],[key1:value5, key2:value6]] key: key值
value: 字典[key1:value1, key2:value2] 对应的模型
*/ @objc public protocol DictModelProtocol{
static func customClassMapping() -> [String: String]?
} extension NSObject{ //dict: 要进行转换的字典
class func objectWithKeyValues(dict: NSDictionary)->AnyObject?{
if HEFoundation.isClassFromFoundation(self) {
print("只有自定义模型类才可以字典转模型")
assert(true)
return nil
} let obj:AnyObject = self.init()
var cls:AnyClass = self.classForCoder() //当前类的类型 while("NSObject" != "\(cls)"){
var count:UInt32 = 0
let properties = class_copyPropertyList(cls, &count) //获取属性列表
for i in 0..<count{ let property = properties[Int(i)] //获取模型中的某一个属性 let propertyType = String.fromCString(property_getAttributes(property))! //属性类型 let propertyKey = String.fromCString(property_getName(property))! //属性名称
if propertyKey == "description"{ continue } //description是Foundation中的计算型属性,是实例的描述信息 var value:AnyObject! = dict[propertyKey] //取得字典中的值
if value == nil {continue} let valueType = "\(value.classForCoder)" //字典中保存的值得类型
if valueType == "NSDictionary"{ //1,值是字典。 这个字典要对应一个自定义的模型类并且这个类不是Foundation中定义的类型。
let subModelStr:String! = HEFoundation.getType(propertyType)
if subModelStr == nil{
print("你定义的模型与字典不匹配。 字典中的键\(propertyKey) 对应一个自定义的 模型")
assert(true)
}
if let subModelClass = NSClassFromString(subModelStr){
value = subModelClass.objectWithKeyValues(value as! NSDictionary) //递归
}
}else if valueType == "NSArray"{ //值是数组。 数组中存放字典。 将字典转换成模型。 如果协议中没有定义映射关系,就不做处理 if self.respondsToSelector("customClassMapping") {
if var subModelClassName = cls.customClassMapping()?[propertyKey]{ //子模型的类名称
subModelClassName = HEFoundation.bundlePath+"."+subModelClassName
if let subModelClass = NSClassFromString(subModelClassName){
value = subModelClass.objectArrayWithKeyValuesArray(value as! NSArray);
}
}
} } obj.setValue(value, forKey: propertyKey)
}
free(properties) //释放内存
cls = cls.superclass()! //处理父类
}
return obj
} /**
将字典数组转换成模型数组
array: 要转换的数组, 数组中包含的字典所对应的模型类就是 调用这个类方法的类 当数组中嵌套数组, 内部的数组包含字典,cls就是内部数组中的字典对应的模型
*/
class func objectArrayWithKeyValuesArray(array: NSArray)->NSArray?{
if array.count == 0{
return nil
}
var result = [AnyObject]()
for item in array{
let type = "\(item.classForCoder)"
if type == "NSDictionary"{
if let model = objectWithKeyValues(item as! NSDictionary){
result.append(model)
}
}else if type == "NSArray"{
if let model = objectArrayWithKeyValuesArray(item as! NSArray){
result.append(model)
}
}else{
result.append(item)
}
}
if result.count==0{
return nil
}else{
return result
}
}
}

模型-->字典

//
// HE_Model2Dict.swift
// HEExtention
//
// Created by 贺俊孟 on 16/4/27.
// Copyright © 2016年 贺俊孟. All rights reserved.
// 模型传字典 import Foundation extension NSObject{
var keyValues:[String:AnyObject]?{ //获取一个模型对应的字典
get{
var result = [String: AnyObject]() //保存结果
var classType:AnyClass = self.classForCoder
while("NSObject" != "\(classType)" ){
var count:UInt32 = 0
let properties = class_copyPropertyList(classType, &count)
for i in 0..<count{
let property = properties[Int(i)]
let propertyKey = String.fromCString(property_getName(property))! //模型中属性名称
let propertyType = String.fromCString(property_getAttributes(property))! //模型中属性类型 if "description" == propertyKey{ continue } //描述,不是属性 let tempValue:AnyObject! = self.valueForKey(propertyKey)
if tempValue == nil { continue } if let _ = HEFoundation.getType(propertyType) { //1,自定义的类
result[propertyKey] = tempValue.keyValues
}else if (propertyType.containsString("NSArray")){ //2, 数组, 将数组中的模型转成字典
result[propertyKey] = tempValue.keyValuesArray //3, 基本数据
}else{
result[propertyKey] = tempValue
}
}
free(properties)
classType = classType.superclass()!
}
if result.count == 0{
return nil
}else{
return result
} }
}
} extension NSArray{ //数组的拓展
var keyValuesArray:[AnyObject]?{
get{
var result = [AnyObject]()
for item in self{
if !HEFoundation.isClassFromFoundation(item.classForCoder){ //1,自定义的类
let subKeyValues:[String:AnyObject]! = item.keyValues
if subKeyValues == nil {continue}
result.append(subKeyValues)
}else if item.classForCoder == NSArray.classForCoder(){ //2, 如果item 是数组
let subKeyValues:[AnyObject]! = item.keyValuesArray
if subKeyValues == nil {continue}
result.append(subKeyValues)
}else{ //3, 基本数据类型
result.append(item)
}
}
if result.count == 0{
return nil
}else{
return result
} }
}
}

辅助类

//
// HEFoundation.swift
// HEExtention
//
// Created by 贺俊孟 on 16/4/27.
// Copyright © 2016年 贺俊孟. All rights reserved. import Foundation class HEFoundation { static let set = NSSet(array: [
NSURL.classForCoder(),
NSDate.classForCoder(),
NSValue.classForCoder(),
NSData.classForCoder(),
NSError.classForCoder(),
NSArray.classForCoder(),
NSDictionary.classForCoder(),
NSString.classForCoder(),
NSAttributedString.classForCoder()
])
static let bundlePath = NSBundle.mainBundle().infoDictionary!["CFBundleExecutable"] as! String /*** 判断某个类是否是 Foundation中自带的类 */
class func isClassFromFoundation(c:AnyClass)->Bool {
var result = false
if c == NSObject.classForCoder(){
result = true
}else{
set.enumerateObjectsUsingBlock({ (foundation, stop) -> Void in
if c.isSubclassOfClass(foundation as! AnyClass) {
result = true
stop.initialize(true)
}
})
}
return result
} /** 很据属性信息, 获得自定义类的 类名*/
/**
let propertyType = String.fromCString(property_getAttributes(property))! 获取属性类型
到这个属性的类型是自定义的类时, 会得到下面的格式: T+@+"+..+工程的名字+数字+类名+"+,+其他,
而我们想要的只是类名,所以要修改这个字符串
*/
class func getType(var code:String)->String?{ if !code.containsString(bundlePath){ //不是自定义类
return nil
}
code = code.componentsSeparatedByString("\"")[1]
if let range = code.rangeOfString(bundlePath){
code = code.substringFromIndex(range.endIndex)
var numStr = "" //类名前面的数字
for c:Character in code.characters{
if c <= "9" && c >= "0"{
numStr+=String(c)
}
}
if let numRange = code.rangeOfString(numStr){
code = code.substringFromIndex(numRange.endIndex)
}
return bundlePath+"."+code
}
return nil
}
}

Swift 字典模型互转总结的更多相关文章

  1. IOS 字典模型互转框架 MJExtension

    IOS 字典模型互转框架 MJExtension   能做什么? MJExtension是一套字典和模型之间互相转换的超轻量级框架 MJExtension能完成的功能 字典(JSON) -->  ...

  2. swift字典集合-备

    Swift字典表示一种非常复杂的集合,允许按照某个键来访问元素.字典是由两部分集合构成的,一个是键(key)集合,一个是值(value)集合.键集合是不能有重复元素的,而值集合是可以重复的,键和值是成 ...

  3. Swift 字典 Dictionary基本用法

    import UIKit /* 字典的介绍 1.字典允许按照某个键访问元素 2.字典是由两部分组成, 一个键(key)集合, 一个是值(value)集合 3.键集合是不能有重复的元素, 值集合可以有重 ...

  4. MVC中Model BLL层Model模型互转

    MVC中Model BLL层Model模型互转 一. 模型通常可以做2种:充血模型和失血模型,一般做法是模型就是模型,不具备方法来操作,只具有属性,这种叫做失血模型(可能不准确):具备对模型一定的简单 ...

  5. 使用Runtime的objc_msgSend实现模型和字典的互转

    一.介绍 模型转字典,字典转模型,这是开发中最基本的功能.系统类中提供了一个setValuesForKeysWithDictionary方法来实现字典转模型,至于模型转字典,这个就需要使用runtim ...

  6. swift - 字典和集合

    p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 13.0px Menlo; color: #4dbf56 } p.p2 { margin: 0.0px 0. ...

  7. JsonString,字典,模型之间相互转换

    NSData转字符串 [NSString alloc] initWithData: encoding:] 模型转字典 attInfo.keyValues 字典转模型 ZTEOutputInfo *ou ...

  8. MVC MVVM Knockout viewmodel 提交 完整过程,包含序列化 JSON 和 字典模型绑定

    //JSON 通用操作------------------------------------------------------------------------------using Syste ...

  9. Swift字典

    字典初始化 基本语法: [key 1: value 1, key 2: value 2, key 3: value 3] var   airports:    Dictionary<String ...

随机推荐

  1. 【UWP】仅在TextBlock文本溢出时显示Tooltip

    前言 这是我今天在回答SO问题时偶然遇到的,觉得可能还比较通用,就记录下来以供参考. 通常,我们使用ToolTip最简单的方式是这样: <TextBlock Text="Test&qu ...

  2. 037.[转] springboot 配置多个数据源

    1.在application.properties文件 配置两个数据源 #默认使用 tomcat-jdbc spring.datasource.type=org.apache.tomcat.jdbc. ...

  3. HTTPS混合内容解析

    什么是HTTPS混合内容 我们可能会有这样的经验,当我们通过HTTPS访问一个网站的时候,突然有提示:“本页面包含有不安全的内容”.这个时候会询问是否显示“不安全的内容”,这个时候,就是遇到了有混合内 ...

  4. 02-Node.js学习笔记-系统模块fs文件操作

    2.1.什么是系统模块 Node 运行环境提供的API,因为这些API都是以模块化的方式进行开发的,所有我们又称Node运行环境提供的API为系统模块 3.1系统模块fs文件操作 //f :file ...

  5. OSPF和ACL的应用

    1.创建拓扑图 2.配置基本网络 3.配置OSPF (1)在R1上配置 (2)在R2上配置 (3)在R3上配置 (4)在IT上配置 4.配置ACL (1)在R3上配置    (2)在R2上配置 (3) ...

  6. poj 3070 矩阵计算Fibonacci

    地址 http://poj.org/problem?id=3070 大意是输入一个数字 输出位于Fibonacci数列该位置的数字模10000的结果 由于n比较大 0 ≤ n ≤ 1,000,000, ...

  7. Java Web 学习(5) —— Spring MVC 之数据绑定

    Spring MVC 之数据绑定 数据绑定是将用户输入绑定到领域模型的一种特性. Http 请求传递的数据为 String 类型,通过数据绑定,可以将数据填充为不同类型的对象属性. 基本类型绑定 @R ...

  8. LG5202 「USACO2019JAN」Redistricting 动态规划+堆/单调队列优化

    问题描述 LG5202 题解 \[opt[i]=xx+(cnt[i]-cnt[yy]<=0)\] 发现\(cnt[i]-cnt[yy] <= 0\)只能有两种取值 于是直接堆优化即可 \( ...

  9. 安装office2010出现的一些问题

    在往新的台式机安装office2010时,因为一些原因没有使用虚拟光驱安装,而是解压后安装,安装过程中提示错误,无法找到excel.zh-cn中的msi文件,但是查看安装目录文件发现是有的. 后续尝试 ...

  10. 【转】Java 浅拷贝和深拷贝的理解和实现方式

    Java中的对象拷贝(Object Copy)指的是将一个对象的所有属性(成员变量)拷贝到另一个有着相同类类型的对象中去.举例说明:比如,对象A和对象B都属于类S,具有属性a和b.那么对对象A进行拷贝 ...