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

转载原文地址

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

分析用例

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

class TestASwiftClass {
    var aBoll : Bool = true
    var aInt : Int = 0
    var aFloat : Float = 123.45
    var aDouble : Double = 1234.567
    var  aString :String = "abc"
    var aObject : AnyObject! = nil
    func testReturnVoidWithaId(aId : UIView) {
        print("TestASwiftClass.testReturnVoidWithaId")
    }
}

class testSwiftVC : UIViewController {
    var aBoll : Bool = true
    var aInt : Int = 0
    var aFloat : Float = 123.45
    var aDouble : Double = 1234.567
    var  aString :String = "abc"
    var aObject : AnyObject! = nil
    override func viewDidLoad() {
        super.viewDidLoad()
    }
    override func viewDidAppear(animated: Bool) {
        super.viewDidAppear(animated)
    }
    func testReturnVoidWithaId(aId : UIView) {
        print("testSwiftVC.testReturnVoidWithaId")
    }
    func testReturnVoidWithBool(aBool : Bool , aInteger : Int , aFloat : Float , aString : String , aObject : AnyObject) {
        print("testSwiftVC.testReturnVoidWithBool")
    }
    func testReturnTuple(aBool : Bool , aInteger : Int , aFloat : Float) -> (Bool , Int , Float){
        print("testSwiftVC.testReturnTuple")
        return (aBool,aInteger,aFloat)
    }
    func testReturnVoidWithCharacter(aCharacter : Character ) {
        print("testSwiftVC.testReturnVoidWithCharacter")
    }
    func tableView(table : UITableView , numberOfRowsInSection section : Int) -> Int {
        print("testSwiftVC.tableView(table : UITableView , numberOfRowsInSection section : Int) -> Int")
        return 20
    }
}
方法、属性

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

func showClsRuntime (cls : AnyClass)  {
    print("start methodList")
    var  methodNum : UInt32 = 0
    let methodList = class_copyMethodList(cls, &methodNum)
    for index in 0..<numericCast(methodNum) {
        let method : Method = methodList[index]
        print(String(UTF8String: method_getTypeEncoding(method)))
        print(String(UTF8String: method_copyReturnType(method)))
        print(String(_sel: method_getName(method)))
    }
    free(methodList)
    print("end methodList")
    
    print("start propertyList")
    
    var propertyNum : UInt32 = 0
    let propertyList = class_copyPropertyList(cls, &propertyNum)
    for index in 0..<numericCast(propertyNum) {
        let property : objc_property_t = propertyList[index]
        print(String(UTF8String: property_getName(property)))
        print(String(UTF8String: property_getAttributes(property)))
    }
    free(propertyList)
    print("end propertyList")
    
}

调用showClsRuntime的代码如下:

let aSwiftClass : TestASwiftClass = TestASwiftClass()
        showClsRuntime(object_getClass(aSwiftClass))
        print("\n\n\n")
        showClsRuntime(object_getClass(self))

看看我们得到什么结果?

***start methodList
---end methodList


***start propertyList
---end propertyList




***start methodList
Optional("B16@0:8")
Optional("B")
aBoll
Optional("v20@0:8B16")
Optional("v")
setABoll:
Optional("q16@0:8")
Optional("q")
aInt
Optional("v24@0:8q16")
Optional("v")
setAInt:
Optional("f16@0:8")
Optional("f")
aFloat
Optional("v20@0:8f16")
Optional("v")
setAFloat:
Optional("d16@0:8")
Optional("d")
aDouble
Optional("v24@0:8d16")
Optional("v")
setADouble:
Optional("@16@0:8")
Optional("@")
aString
Optional("v24@0:8@16")
Optional("v")
setAString:
Optional("@16@0:8")
Optional("@")
aObject
Optional("v24@0:8@16")
Optional("v")
setAObject:
Optional("v24@0:8@16")
Optional("v")
testReturnVoidWithaId:
Optional("v48@0:8B16q20f28@32@40")
Optional("v")
testReturnVoidWithBool:aInteger:aFloat:aString:aObject:
Optional("q32@0:8@16q24")
Optional("q")
tableView:numberOfRowsInSection:
Optional("v16@0:8")
Optional("v")
didReceiveMemoryWarning
Optional("@32@0:8@16@24")
Optional("@")
initWithNibName:bundle:
Optional("v16@0:8")
Optional("v")
viewDidLoad
Optional("v20@0:8B16")
Optional("v")
viewDidAppear:
Optional("@?")
Optional("@?")
.cxx_destruct
Optional("@24@0:8@16")
Optional("@")
initWithCoder:
---end methodList


***start propertyList
Optional("aBoll")
Optional("TB,N,VaBoll")
Optional("aInt")
Optional("Tq,N,VaInt")
Optional("aFloat")
Optional("Tf,N,VaFloat")
Optional("aDouble")
Optional("Td,N,VaDouble")
Optional("aString")
Optional("T@\"NSString\",N,C,VaString")
Optional("aObject")
Optional("T@,N,&,VaObject")
---end propertyList

对于纯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的代码如下:

///Method Swizzeing runtime动态替换方法
func methodSwizze(cls : AnyClass,originalSelector : Selector , swizzeSelector : Selector)  {
    let originalMethod = class_getInstanceMethod(cls, originalSelector)
    let swizzeMethod = class_getInstanceMethod(cls, swizzeSelector)
    
    let didAddMethod = class_addMethod(cls, originalSelector, method_getImplementation(swizzeMethod), method_getTypeEncoding(swizzeMethod))
    
    if didAddMethod {
        class_replaceMethod(cls, swizzeSelector, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod))
    }else {
        method_exchangeImplementations(originalMethod, swizzeMethod)
    }
    
}

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

  override func viewDidLoad() {
        super.viewDidLoad()
//        let aSwiftClass : TestASwiftClass = TestASwiftClass()
//        showClsRuntime(object_getClass(aSwiftClass))
//        print("\n\n\n")
//        showClsRuntime(object_getClass(self))
        
        methodSwizze(object_getClass(self), originalSelector: Selector("viewDidAppear:"), swizzeSelector: Selector("sz_viewDidApper:"))
        methodSwizze(object_getClass(self), originalSelector: Selector("testReturnVoidWithaId:"), swizzeSelector: Selector("sz_testReturnVoidWithaId:"))
        testReturnVoidWithaId(self.view)
    }
    override func viewDidAppear(animated: Bool) {
        super.viewDidAppear(animated)
        print("F:\(__FUNCTION__) L:\(__LINE__))")
    }
    func testReturnVoidWithaId(aId : UIView) {
        print("testSwiftVC.testReturnVoidWithaId------F:\(__FUNCTION__) L:\(__LINE__))")
    }
    
    func sz_viewDidApper(animated : Bool) {
        super.viewDidAppear(animated)
        print("SZ------F:\(__FUNCTION__) L:\(__LINE__))")
        
    }
    
    func sz_testReturnVoidWithaId(aId : UIView) {
        print("SZ:testSwiftVC.testReturnVoidWithaId------F:\(__FUNCTION__) L:\(__LINE__))")
    }
    

打印的日志为:

testSwiftVC.testReturnVoidWithaId------F:testReturnVoidWithaId L:90)
SZ------F:sz_viewDidApper L:95)

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

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

屏幕快照 2016-04-20 下午3.13.31.png

可以看到区别,调用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 试试,如:

class TestASwiftClass {
   @objc  var aBoll : Bool = true
   @objc var aInt : Int = 0
   @objc var aFloat : Float = 123.45
   @objc var aDouble : Double = 1234.567
   @objc var  aString :String = "abc"
   @objc var aObject : AnyObject! = nil
   @objc func testReturnVoidWithaId(aId : UIView) {
        print("TestASwiftClass.testReturnVoidWithaId")
    }
}

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

***start methodList
Optional("B16@0:8")
Optional("B")
aBoll
Optional("v20@0:8B16")
Optional("v")
setABoll:
Optional("q16@0:8")
Optional("q")
aInt
Optional("v24@0:8q16")
Optional("v")
setAInt:
Optional("f16@0:8")
Optional("f")
aFloat
Optional("v20@0:8f16")
Optional("v")
setAFloat:
Optional("d16@0:8")
Optional("d")
aDouble
Optional("v24@0:8d16")
Optional("v")
setADouble:
Optional("@16@0:8")
Optional("@")
aString
Optional("v24@0:8@16")
Optional("v")
setAString:
Optional("@16@0:8")
Optional("@")
aObject
Optional("v24@0:8@16")
Optional("v")
setAObject:
Optional("v24@0:8@16")
Optional("v")
testReturnVoidWithaId:
---end methodList
dynamic

文档里还有一句说明:

加了@objc标识的方法、属性无法保证都会被运行时调用,因为Swift会做静态优化。要想完全被动态调用,必须使用dynamic修饰。使用dynamic修饰将会隐式的加上@objc标识。

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

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

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

屏幕快照 2016-04-20 下午3.19.25.png

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

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

屏幕快照 2016-04-20 下午3.19.39.png
Objective-C获取Swift runtime信息

在Objective-c代码里使用objc_getClass("TestSwiftVC");会发现返回值为空,这是为什么?Swift代码中的TestSwiftVC类,在OC中还是这个名字吗?
我们初始化一个对象,并断点和打印看看,如下图:

屏幕快照 2016-04-20 下午3.21.44.png

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

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

屏幕快照 2016-04-20 下午3.24.01.png
Objective-C替换Swift函数

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

![Upload 屏幕快照 2016-04-20 下午3.24.10.png failed. Please try again.]

运行之后我们得到结果

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中会有模块前缀
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容

  • 转载自:移动开发前线 Swift是苹果2014年发布的编程开发语言,可与Objective-C共同运行于Mac O...
    MichleMin阅读 983评论 0 0
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,614评论 4 59
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,635评论 0 9
  • 01 “咚咚,咚咚,咚咚……” 听,这是人类的心跳声。亡灵们,你们永远都不会有,是什么样的怨恨,让你们没有心跳却又...
    半朽阅读 685评论 4 18
  • 前段时间一直在看蔡澜老先生的一本书《今天也要好好吃饭》。 有一段是描写怎样吃羊肉的。我顿时,觉得自己吃羊肉是一件多...
    医知肤胡老师阅读 373评论 0 2