Swift Runtime动态性分析:还像OC Runtime一样吗?

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友 微信微信
查看查看323 回复回复1 收藏收藏 分享淘帖 转播转播 分享分享 微信
查看: 323|回复: 1
收起左侧

Swift Runtime动态性分析:还像OC Runtime一样吗?

[复制链接]
cocoaswift 发表于 2016-6-21 19:44:03 | 显示全部楼层 |阅读模式
快来登录
获取优质的苹果资讯内容
收藏热门的iOS等技术干货
拷贝下载Swift Demo源代码
订阅梳理好了的知识点专辑

Swift是苹果2014年发布的编程开发语言,可与Objective-C共同运行于Mac OS和iOS平台,用于搭建基于苹果平台的应用程序。Swift已经开源,目前最新版本为3.0,本文是对2.2版本语法而言的。我们知道Objective-C是具有动态性的,能够通过runtime API调用和替换任意方法,那Swift也具有这些动态性吗?

Swift Runtime动态性分析:还像OC Runtime一样吗? 0

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 0





1、分析用例

我们拿一个纯Swift类和一个继承自NSObject的类来做分析,这两个类里包含尽量多的Swift的类型比如Character、String、AnyObject、Tuple。

代码如下:

Swift Runtime动态性分析:还像OC Runtime一样吗? 1

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 1





2、方法、属性

动态性比较重要的一点就是能够拿到某个类所有的方法、属性,我们使用如下代码来打印方法和属性列表。

Swift Runtime动态性分析:还像OC Runtime一样吗? 2

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 2


调用showClsRuntime的代码如下:

[Swift] 纯文本查看 复制代码
let aSwiftClass:TestASwiftClass = TestASwiftClass();
showClsRuntime(object_getClass(aSwiftClass));
print("\n");
showClsRuntime(object_getClass(self));


看看我们得到什么结果?

Swift Runtime动态性分析:还像OC Runtime一样吗? 3

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 3


* 对于纯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获取了。




3、Method Swizzling

动态性最常用的就是方法替换(Method Swizzling),将类的某个方法替换成自定义的方法,从而达到hook的作用。

  • 对于纯Swift类(如TestASwiftClass)来说,无法通过objc runtime替换方法,因为由上面的测试可知拿不到这些方法、属性
  • 对于继承自NSObject类(如TestSwiftVC)来说,无法通过runtime获取到的方法肯定没法替换了。那能通过runtime获取到的方法就都能被替换吗?我们测一把。


Method Swizzling的代码如下

Swift Runtime动态性分析:还像OC Runtime一样吗? 4

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 4


我们替换两个可以被runtime获取到的方法:viewDidAppear和testReturnVoidWithaId

Swift Runtime动态性分析:还像OC Runtime一样吗? 5

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 5


打印的日志为

[Swift] 纯文本查看 复制代码
F:testReturnVoidWithaId L:50
F:sz_viewDidAppear L:46


说明viewDidAppear已经被替换,但是testReturnVoidWithaId却没有被替换,这是为何?

我们在方法里打个断点看看,如图:

Swift Runtime动态性分析:还像OC Runtime一样吗? 6

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 6


Swift Runtime动态性分析:还像OC Runtime一样吗? 7

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 7


可以看到区别,调用sz_viewDidAppear栈的前一帧为@objc TestSwiftVC.sz_viewDidAppear(Bool) -> ()有个@objc标识,而调用testReturnVoidWithaId则没有此标识。

@objc用来做什么的?与动态性有关吗?




4、@objc

找到官方文档读读。

可以知道@objc是用来将Swift的API导出给Objective-C和Objective-C runtime使用的,如果你的类继承自Objective-c的类(如NSObject)将会自动被编译器插入@objc标识。

我们在把TestASwiftClass(纯Swift类)的方法、属性前都加个@objc 试试,如图:

Swift Runtime动态性分析:还像OC Runtime一样吗? 8

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 8


查看日志可以发现加了@objc的方法、属性均可以被runtime获取到了。

Swift Runtime动态性分析:还像OC Runtime一样吗? 9

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 9





5、dynamic

文档里还有一句说明:

加了@objc标识的方法、属性无法保证都会被运行时调用,

因为Swift会做静态优化。要想完全被动态调用,必须使用dynamic修饰。

使用dynamic修饰将会隐式的加上@objc标识

这也就解释了为什么testReturnVoidWithaId无法被替换,因为写在Swift里的代码直接被编译优化成静态调用了。

而viewDidAppear是继承Objective-C类获得的方法,本身就被修饰为dynamic,所以能被动态替换。

我们把TestSwiftVC方法前加上dynamic再测一把,如图:

Swift Runtime动态性分析:还像OC Runtime一样吗? 10

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 10


从堆栈也可以看出,方法的调用前增加了@objc标识,testReturnVoidWithaId方法被替换成功了。

同样的做法,我们把TestASwiftClass的方法和属性也都加上dynamic修饰,做Method Swizzling,同样获得成功,如图:

Swift Runtime动态性分析:还像OC Runtime一样吗? 11

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 11





6、Objective-C获取Swift runtime信息

在Objective-C代码里使用objc_getClass("TestSwiftVC");会发现返回值为空,这是为什么?Swift代码中的TestSwiftVC类,在OC中还是这个名字吗?

我们初始化一个对象,并断点和打印看看,如下图:

Swift Runtime动态性分析:还像OC Runtime一样吗? 12

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 12


可以看到Swift中的TestSwiftVC类在OC中的类名已经变成TestSwift.TestSwiftVC,即规则为SWIFT_MODULE_NAME.类名称,在普通源码项目里SWIFT_MODULE_NAME即为ProductName,在打好的Cocoa Touch Framework里为则为导出的包名。

所以要想从Objective-c中获取Swift类的runtime信息得这样写:

[Swift] 纯文本查看 复制代码
id cls = objc_getClass("TestSwift.TestASwiftClass");
showClsRuntime(cls);
id cls2 = objc_getClass("TestSwift.TestSwiftVC");
showClsRuntime(cls2);





7、Objective-C替换Swift函数

给TestSwiftVC和TestASwiftClass的testReturnVoidWithaId函数加上dynamic修饰,然后我们在Objective-C代码里替换为testReturnVoidWithaIdImp函数:

Swift Runtime动态性分析:还像OC Runtime一样吗? 13

Swift Runtime动态性分析:还像OC Runtime一样吗? - 敏捷大拇指 - Swift Runtime动态性分析:还像OC Runtime一样吗? 13


运行之后我们得到结果

[Swift] 纯文本查看 复制代码
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也会打印日志,为什么?留给读者思考)




8、总结

  • 纯Swift类没有动态性,但在方法、属性前添加dynamic修饰可以获得动态性。
  • 继承自NSObject的Swift类,其继承自父类的方法具有动态性,其他自定义方法、属性需要加dynamic修饰才可以获得动态性。
  • 若方法的参数、属性类型为Swift特有、无法映射到Objective-C的类型(如Character、Tuple),则此方法、属性无法添加dynamic修饰(会编译错误)。
  • Swift类在Objective-C中会有模块前缀。






作者:尹峥伟(花名 君展),来自手机淘宝技术团队的资深无线开发工程师,主要负责手机淘宝基础架构研发,github开源库Wax的维护者,微信号yzwlvzxh,微博@君展。

都看到这里了,就把这篇资料推荐给您的好朋友吧,让他们也感受一下。

回帖是一种美德,也是对楼主发帖的尊重和支持。

*声明:敏捷大拇指是全球最大的Swift开发者社区、苹果粉丝家园、智能移动门户,所载内容仅限于传递更多最新信息,并不意味赞同其观点或证实其描述;内容仅供参考,并非绝对正确的建议。本站不对上述信息的真实性、合法性、完整性做出保证;转载请注明来源并加上本站链接,敏捷大拇指将保留所有法律权益。如有疑问或建议,邮件至marketing@swifthumb.com

*联系:微信公众平台:“swifthumb” / 腾讯微博:@swifthumb / 新浪微博:@swifthumb / 官方QQ一群:343549891(满) / 官方QQ二群:245285613 ,需要报上用户名才会被同意进群,请先注册敏捷大拇指

嗯,不错!期待更多好内容,支持一把:
支持敏捷大拇指,用支付宝支付10.24元 支持敏捷大拇指,用微信支付10.24元

评分

参与人数 1金钱 +10 贡献 +10 专家分 +10 收起 理由
Anewczs + 10 + 10 + 10 32个赞!专家给力!

查看全部评分

baddy 发表于 2016-6-23 12:42:13 | 显示全部楼层
Runtime  
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

做任务,领红包。
我要发帖

分享扩散

都看到这里了,就把这资料推荐给您的好朋友吧,让他们也感受一下。
您的每一位朋友访问此永久链接后,您都将获得相应的金钱积分奖励
热门推荐

合作伙伴

Swift小苹果

  • 北京治世天下科技有限公司
  • ©2014-2016 敏捷大拇指
  • 京ICP备14029482号
  • Powered by Discuz! X3.1 Licensed
  • swifthumb Wechat Code
  •   
快速回复 返回顶部 返回列表