一、Swift反射

所谓反射就是可以动态获取类型、成员信息,在运行时可以调用方法、属性等行为的特性。 在使用OC开发时很少强调其反射概念,因为OC的Runtime要比其他语言中的反射强大的多。不过在Swift中并不提倡使用Runtime,而是像其他语言一样使用反射(Reflect),即使目前Swift中的反射功能还比较弱,只能访问获取类型、成员信息。

Swift的反射机制是基于一个叫Mirror的结构体来实现的。你为具体的实例创建一个Mirror对象,然后就可以通过它查询这个实例

Mirror结构体常用属性:
  • subjectType:对象类型
  • children:反射对象的属性集合
  • displayStyle:反射对象展示类型
下面来简单介绍下Mirror的使用:
//定义一个类来进行测试
class Person {
var name: String?
var age: Int = 0
}
//创建一个对象并初始化
let p = Person()
p.name = "小强"
p.age = 13 //1. 创建对象的反射,获取对象类型
let mirror: Mirror = Mirror(reflecting:p)
print("获取对象类型\(mirror.subjectType)")
// 打印出:获取对象类型Person //2. 获取对象属性名以及对应的值
for p in mirror.children {
let propertyNameString = p.label! //属性名使用!,因为label是optional类型
let value = p.value //属性的值
print("\(propertyNameString)的值为\(value)")
}
/* 打印:
name的值为Optional("小强")
age的值为13
*/ //3. 获取指定索引下的属性类型
let children = mirror.children
let p0 = children.startIndex.advancedBy(0) //获取name属性的位置索引
let p0Mirror = Mirror(reflecting: children[p0].value) //name的反射
print("获取属性name的类型为\(p0Mirror.subjectType)")
//打印:获取属性name的类型为Optional //4. 遍历获取对象所有动态的属性类型
for p in mirror.children {
let propertyNameString = p.label!
let value = p.value
let vMirror = Mirror(reflecting: value) //通过值来创建属性的反射
print("属性\(propertyNameString)类型为\(vMirror.subjectType)")
}
/* 打印:
属性name类型为Optional
属性age类型为Int
*/

反射的应用场景现在还比较狭窄,因为功能还不够完善,我提供一个比较常见的反射应用场景,那就是自定义类模型转字典

以下就是自定义类模型转字典实例
//自定义用户类
class User {
var name:String = "" //姓名
var nickname:String? //昵称
var age:Int? //年龄
var emails:[String]? //邮件地址
var tels:[Telephone]? //电话
}
//电话结构体
struct Telephone {
var title:String //电话标题
var number:String //电话号码
}
//自定义一个JSON协议
protocol JSON {
func toJSONModel() -> Any?
}
//扩展协议方法,实现一个通用的toJSONModel方法(反射实现)
extension JSON {
//将模型数据转成可用的字典数据,Any表示任何类型,除了方法类型
func toJSONModel() -> Any? {
//根据实例创建反射结构体Mirror
let mirror = Mirror(reflecting: self)
if mirror.children.count > 0 {
//创建一个空字典,用于后面添加键值对
var result: [String:Any] = [:]
//遍历实例的所有属性集合
for children in mirror.children {
let propertyNameString = children.label!
let value = children.value
//判断value的类型是否遵循JSON协议,进行深度递归调用
if let jsonValue = value as? JSON {
result[propertyNameString] = jsonValue.toJSONModel()
}
}
return result
}
return self
}
}
//扩展可选类型,使其遵循JSON协议,可选类型值为nil时,不转化进字典中
extension Optional: JSON {
//可选类型重写toJSONModel()方法
func toJSONModel() -> Any? {
if let x = self {
if let value = x as? JSON {
return value.toJSONModel()
}
}
return nil
}
}
//扩展两个自定义类型,使其遵循JSON协议
extension User: JSON { }
extension Telephone: JSON { }
//扩展Swift的基本数据类型,使其遵循JSON协议
extension String: JSON { }
extension Int: JSON { }
extension Bool: JSON { }
extension Dictionary: JSON { }
extension Array: JSON { } //创建一个User实例对象模型
let user1 = User()
user1.name = "hangge"
user1.age = 100
user1.emails = ["hangge@hangge.com","system@hangge.com"]
//添加电话
let tel1 = Telephone(title: "手机", number: "123456")
let tel2 = Telephone(title: "公司座机", number: "001-0358")
user1.tels = [tel1, tel2]
//模型转字典
if let model = user1.toJSONModel() {
print(model)
}
/* 打印:【以下打印经过排版,正式的打印是紧凑的】
[
"tels": [
"[1]": [
"title": "公司座机",
"number": "001-0358"
],
"[0]": [
"title": "手机",
"number": "123456"
]
],
"name": "hangge",
"emails": [
"[1]": "system@hangge.com",
"[0]": "hangge@hangge.com"
],
"age": 100
]
*/

iOS学习笔记48-Swift(八)反射的更多相关文章

  1. Android(java)学习笔记48:通过反射获得带参构造方法并且使用

    1. 反射获得带参构造方法并且使用: (1)获取字节码文件对象       Class c = Class.forName("cn.itcast_01.Person"); (2)获 ...

  2. iOS学习笔记-精华整理

    iOS学习笔记总结整理 一.内存管理情况 1- autorelease,当用户的代码在持续运行时,自动释放池是不会被销毁的,这段时间内用户可以安全地使用自动释放的对象.当用户的代码运行告一段 落,开始 ...

  3. iOS学习笔记总结整理

    来源:http://mobile.51cto.com/iphone-386851_all.htm 学习IOS开发这对于一个初学者来说,是一件非常挠头的事情.其实学习IOS开发无外乎平时的积累与总结.下 ...

  4. 【Unity Shaders】学习笔记——SurfaceShader(八)生成立方图

    [Unity Shaders]学习笔记——SurfaceShader(八)生成立方图 转载请注明出处:http://www.cnblogs.com/-867259206/p/5630261.html ...

  5. iOS学习笔记-自定义过渡动画

    代码地址如下:http://www.demodashi.com/demo/11678.html 这篇笔记翻译自raywenderlick网站的过渡动画的一篇文章,原文用的swift,由于考虑到swif ...

  6. iOS学习笔记——AutoLayout的约束

    iOS学习笔记——AutoLayout约束 之前在开发iOS app时一直以为苹果的布局是绝对布局,在IB中拖拉控件运行或者直接使用代码去调整控件都会发上一些不尽人意的结果,后来发现iOS在引入了Au ...

  7. IOS学习笔记25—HTTP操作之ASIHTTPRequest

    IOS学习笔记25—HTTP操作之ASIHTTPRequest 分类: iOS2012-08-12 10:04 7734人阅读 评论(3) 收藏 举报 iosios5网络wrapper框架新浪微博 A ...

  8. IOS学习笔记之关键词@dynamic

    IOS学习笔记之关键词@dynamic @dynamic这个关键词,通常是用不到的. 它与@synthesize的区别在于: 使用@synthesize编译器会确实的产生getter和setter方法 ...

  9. iOS学习笔记10-UIView动画

    上次学习了iOS学习笔记09-核心动画CoreAnimation,这次继续学习动画,上次使用的CoreAnimation很多人感觉使用起来很繁琐,有没有更加方便的动画效果实现呢?答案是有的,那就是UI ...

  10. iOS学习笔记之Category

    iOS学习笔记之Category 写在前面 Category是类别(也称为类目或范畴),使用Category,程序员可以为任何已有的类添加方法.使用类别可以对框架提供的类(无法获取源码,不能直接修改) ...

随机推荐

  1. java.lang.NoClassDefFoundError: javax/servlet/jsp/jstl/core/Config

    今天写SpringMvc时,遇到这样一个问题: java.lang.NoClassDefFoundError: javax/servlet/jsp/jstl/core/Config at org.sp ...

  2. css设置禁止文字被选中

    // 禁止文字被鼠标选中 moz-user-select: -moz-none; -moz-user-select: none; -o-user-select:none; -khtml-user-se ...

  3. CentOS lvm

    1.创建PVpvcreate /dev/sdb /dev/sdc或pvcreate /dev/sdb1 /dev/sdc1 2.查看PVpvdisplay 3.创建VGvgcreate vgdata ...

  4. 学习笔记(五): Feature Crosses

    目录 Feature Crosses Encoding Nonlinearity Kinds of feature crosses Glossay Crossing One-Hot Vectors P ...

  5. Codevs1033 蚯蚓的游戏

    题目描述 Description 在一块梯形田地上,一群蚯蚓在做收集食物游戏.蚯蚓们把梯形田地上的食物堆积整理如下: a(1,1)  a(1,2)…a(1,m) a(2,1)  a(2,2)  a(2 ...

  6. 【线性基合并 树链剖分】bzoj4568: [Scoi2016]幸运数字

    板子题 Description A 国共有 n 座城市,这些城市由 n-1 条道路相连,使得任意两座城市可以互达,且路径唯一.每座城市都有一个 幸运数字,以纪念碑的形式矗立在这座城市的正中心,作为城市 ...

  7. 【期望dp】bzoj4832: [Lydsy1704月赛]抵制克苏恩

    这个题面怎么这么歧义…… Description 小Q同学现在沉迷炉石传说不能自拔.他发现一张名为克苏恩的牌很不公平.如果你不玩炉石传说,不必担心,小Q 同学会告诉你所有相关的细节.炉石传说是这样的一 ...

  8. logback写日志

    https://blog.csdn.net/u010128608/article/details/76618263 https://blog.csdn.net/zhuyucheng123/articl ...

  9. linux时区

    1. UTC时区切换到CST 时区# echo "export TZ='Asia/Shanghai'" >> /etc/profile # cat /etc/profi ...

  10. destoon后台权限-不给客户创始人权限并屏蔽部分功能

    1.根目录下后台入口文件admin.php $_founder = $CFG['founderid'] == $_userid ? $_userid : 0;  //  $CFG['founderid ...