Swift - RunTime(动态性) 问题 浅析
Swift是苹果2014年发布的编程开发语言,可与Objective-C共同运行于Mac OS和iOS平台,用于搭建基于苹果平台的应用程序。Swift已经开源,目前最新版本为2.2。我们知道Objective-C是具有动态性的,能够通过runtime API调用和替换任意方法,那Swift也具有这些动态性吗?
分析用例
我们拿一个纯Swift类
和一个继承自NSObject的类
来做分析,这两个类里包含尽量多的Swift的类型比如Character、String、AnyObject、Tuple。
代码如下:
(点击放大图像)
方法、属性
动态性比较重要的一点就是能够拿到某个类所有的方法、属性,我们使用如下代码来打印方法和属性列表。
(点击放大图像)
调用showClsRuntime的代码如下:
let aSwiftClass:TestASwiftClass = TestASwiftClass(); showClsRuntime(object_getClass(aSwiftClass)); print("\n"); showClsRuntime(object_getClass(self));
看看我们得到什么结果?
(点击放大图像)
* 对于纯Swift的TestASwiftClass来说任何方法、属性都未获取到
* 对于TestSwiftVC来说除testReturnTuple
、
testReturnVoidWithaCharacter
两个方法外,其他的都获取成功了。
这是为什么?
- 纯Swift类的函数调用已经不再是Objective-c的运行时发消息,而是类似C++的vtable,在编译时就确定了调用哪个函数,所以没法通过runtime获取方法、属性。
- TestSwiftVC继承自
UIViewController
,基类为NSObject
,而Swift为了兼容Objective-C,凡是继承自NSObject的类都会保留其动态性,所以我们能通过runtime拿到他的方法。
但为什么testReturnTuple
testReturnVoidWithaCharacter
却又获取不到呢?
从Objective-c的runtime 特性可以知道,所有运行时方法都依赖TypeEncoding,也就是method_getTypeEncoding返回的结果,他指定了方法的参数类型以及在函数调用时参数入栈所要的内存空间,没有这个标识就无法动态的压入参数(比如testReturnVoidWithaId: Optional("v24@0:8@16") Optional("v"),表示此方法参数共需24个字节,返回值为void,第一个参数为id,第二个为selector,第三个为id),而Character和Tuple是Swift特有的,无法映射到OC的类型,更无法用OC的typeEncoding表示,也就没法通过runtime获取了。
Method Swizzling
动态性最常用的就是方法替换(Method Swizzling),将类的某个方法替换成自定义的方法,从而达到hook的作用。
- 对于纯Swift类(如TestASwiftClass)来说,无法通过objc runtime替换方法,因为由上面的测试可知拿不到这些方法、属性
- 对于继承自NSObject类(如TestSwiftVC)来说,无法通过runtime获取到的方法肯定没法替换了。那能通过runtime获取到的方法就都能被替换吗?我们测一把。
Method Swizzling的代码如下
(点击放大图像)
我们替换两个可以被runtime获取到的方法:viewDidAppear
和testReturnVoidWithaId
(点击放大图像)
打印的日志为
F:testReturnVoidWithaId L:50 F:sz_viewDidAppear L:46
说明viewDidAppear已经被替换,但是testReturnVoidWithaId却没有被替换,这是为何?
我们在方法里打个断点看看,如图:
(点击放大图像)
(点击放大图像)
可以看到区别,调用sz_viewDidAppear栈的前一帧为@objc TestSwiftVC.sz_viewDidAppear(Bool) -> ()
有个@objc
标识,而调用testReturnVoidWithaId则没有此标识。
@objc
用来做什么的?与动态性有关吗?
@objc
找到官方文档读读。
可以知道@objc是用来将Swift的API导出给Objective-C和Objective-C runtime使用的,如果你的类继承自Objective-c的类(如NSObject)将会自动被编译器插入@objc标识。
我们在把TestASwiftClass(纯Swift类)的方法、属性前都加个@objc 试试,如图:
(点击放大图像)
查看日志可以发现加了@objc的方法、属性均可以被runtime获取到了。
(点击放大图像)
dynamic
文档里还有一句说明:
加了@objc标识的方法、属性无法保证都会被运行时调用,
因为Swift会做静态优化。要想完全被动态调用,必须使用dynamic修饰。
使用dynamic修饰将会隐式的加上@objc标识
这也就解释了为什么testReturnVoidWithaId无法被替换,因为写在Swift里的代码直接被编译优化成静态调用了。
而viewDidAppear是继承Objective-C类获得的方法,本身就被修饰为dynamic,所以能被动态替换。
我们把TestSwiftVC方法前加上dynamic再测一把,如图:
(点击放大图像)
从堆栈也可以看出,方法的调用前增加了@objc标识,testReturnVoidWithaId方法被替换成功了。
同样的做法,我们把TestASwiftClass的方法和属性也都加上dynamic修饰,做Method Swizzling,同样获得成功,如图
(点击放大图像)
Objective-C获取Swift runtime信息
在Objective-c代码里使用objc_getClass("TestSwiftVC");
会发现返回值为空,这是为什么?Swift代码中的TestSwiftVC类,在OC中还是这个名字吗?
我们初始化一个对象,并断点和打印看看,如下图:
(点击放大图像)
可以看到Swift中的TestSwiftVC类在OC中的类名已经变成TestSwift.TestSwiftVC
,即规则为SWIFT_MODULE_NAME.类名称
,在普通源码项目里SWIFT_MODULE_NAME即为ProductName,在打好的Cocoa Touch Framework里为则为导出的包名。
所以要想从Objective-c中获取Swift类的runtime信息得这样写:
id cls = objc_getClass("TestSwift.TestASwiftClass"); showClsRuntime(cls); id cls2 = objc_getClass("TestSwift.TestSwiftVC"); showClsRuntime(cls2);
Objective-C替换Swift函数
给TestSwiftVC和TestASwiftClass的testReturnVoidWithaId
函数加上dynamic修饰,然后我们在Objective-C代码里替换为testReturnVoidWithaIdImp
函数:
(点击放大图像)
运行之后我们得到结果
F:void testReturnVoidWithaIdImp(__strong id, SEL, __strong id) L:20 self=<TestSwift.TestSwiftVC: 0x7fb4e1d148f0> F:void testReturnVoidWithaIdImp(__strong id, SEL, __strong id) L:20 self=TestSwift.TestASwiftClass
说明两者的方法在加上dynamic修饰后,均能在Objective-c里被替换。(TestSwiftVC的testReturnVoidWithaId不加dynamic也会打印日志,为什么?留给读者思考)
总结
- 纯Swift类没有动态性,但在方法、属性前添加dynamic修饰可以获得动态性。
- 继承自NSObject的Swift类,其继承自父类的方法具有动态性,其他自定义方法、属性需要加dynamic修饰才可以获得动态性。
- 若方法的参数、属性类型为Swift特有、无法映射到Objective-C的类型(如Character、Tuple),则此方法、属性无法添加dynamic修饰(会编译错误)
- Swift类在Objective-C中会有模块前缀
转载自:http://www.infoq.com/cn/articles/dynamic-analysis-of-runtime-swift?utm_campaign=rightbar_v2&utm_source=infoq&utm_medium=articles_link&utm_content=link_text
Swift - RunTime(动态性) 问题 浅析相关推荐
- iOS-学习资源看我就够了
iOS学习视频及资料 懒人demo大全更新至4.0版 (oc + swift) 史上最全iOS 开发工具集合 笔试面试 iOS-笔试面试总结 网络通信 <iOS 对 HTTPS 证书链的验证&g ...
- swift int转string_Swift集合类型协议浅析(下)
关注[搜狐技术产品]公众号,第一时间获取技术干货 导读 本篇是Swift集合类型协议浅析系列文章的下篇,在这篇文章中,我们将继续围绕集合类型协议展开讨论,侧重点更多地关注于String相关的周边协议. ...
- swift 组件化_打造完备的iOS组件化方案:如何面向接口进行模块解耦?
作者 | 黑超熊猫zuik,一个修行中的 iOS 开发,喜欢搞点别人没搞过的东西,钻研过逆向工程.VIPER 架构和组件化. 关于组件化的探讨已经有不少了,在之前的文章 iOS VIPER架构实践(三 ...
- iOS热更新解读(三)—— JSPatch 之于 Swift
继承自 NSObject 的 Swift 类 修改属性 新建 Swift 工程 SwiftJSPatch. AppDelegate.swift: // in AppDelegate.swift --- ...
- 手淘 App 如何落地 Swift ?一边探索实践,一边“打怪升级”
嘉宾|姜沂(倾寒) 出品|InfoQ&阿里巴巴新零售淘系技术部 嘉宾简介:姜沂(花名:倾寒),阿里巴巴淘系技术部 iOS 端架构高级无线开发工程师,曾在链家网.美团点评从事 iOS 相关开发, ...
- swift 组件化_京东商城订单模块基于 Swift 的改造方案与实践
ABI Stability & Module Stability 以及Swift优势 2019年Swift 5发布,标志这门语言迎来了一个重大的里程碑.与之前的版本相比除了一些基础语法的改变, ...
- Swift 3.0 beta 6权限访问修改
原文地址: https://kemchenj.github.io/20... title: Swift 3.0 beta 6权限访问修改 date: 2016-08-17 17:14 前两天 swif ...
- Swift之深入解析Xcode13对Swift对象生命周期的优化
在 Xcode13 中,在 Build Setting 中,新增 Optimize Object Lifetimes 编译选项,默认是关闭的,Apple 建议将该选项设置为 YES,打开此优化项,可以 ...
- Swift 在 GAIA 平台云端一体化的探索
作者|姜沂(倾寒) 出品|阿里巴巴新零售淘系技术部 S1 阶段在使用 SwiftUI 编写集团内部使用的 SOT APP 时,有幸参与到 GAIA (FaaS)平台云端一体化的探索,从头到尾实现了一套 ...
最新文章
- 引用类型(一):Object类型
- 学习用Pandas处理分类数据!
- 廖雪峰历时 3 个月打磨出价值 1980 的数据分析教程,终终终于免费啦!
- 数据挖掘:大数据发展的核心驱动力
- linux io100的原因_Unix与Linux的区别与联系
- c语言复杂数据类型存储,C语言基础-复杂数据类型
- java中有效关键字_java中有效的关键字
- IIS Express局域网访问配置
- 第一章 计算机网络概述[知识点+课后习题+练习题]
- Parcelbale接口
- html语言怎么修改form边框,form表单样式案例 定义html表单细边框样式
- SEO之关键词的选择
- 深蓝学院-多传感器融合定位课程学习分享-专栏汇总
- 传奇服务器端回满血文件,单机架设传奇服务器第八课:血过65535
- HDU 6691 Minimum Spanning Trees
- 2017-2018 ACM-ICPC, Asia Daejeon Regional Contest:Gym 101667C
- 推荐系统8——利用社交网络数据推荐
- HP11(HP其他喷头也可) 喷头程序开发项目外包
- 应届生怎么样成为产品经理?
- 别找了,这个命令让你在字符串和十六进制间自由转换
热门文章
- python绝对值函数fabs_Python fabs() 函数
- java 记事本全选_java 编写的记事本程序怎么实现复制 黏贴 剪切 全选的功能 ?...
- java 反射 systemproperties,Android添加自己的系统属性(反射调用SystemProperties)
- oracle建共享dblink,oracle建多个dblink
- arcpy实现空间查询_「实战系列」GP+Roaringbitmap,亿级会员十万级标签毫秒级查询...
- java word转图片tiff_Word 2010中将文档保存为TIFF图片的方法
- VSCode 个性化配置快捷键
- 3.7 测试时的 Batch Norm
- 编程语言排行分析,从2009到2019。
- url动态追加参数_url设置的注意事项有哪些?