Swift底层探索3 - 属性

在 Swift 中属性可以分为两大类:存储属性(Stored Property),计算属性(Computed Property)

1、存储属性

存储属性是一个作为特定类和结构体实例一部分的常量或变量。存储属性要么是变量存储属性 (由 var 关键字引入),要么是常量存储属性(由 let 关键字引入)。

  • let 用来声明常量,常量的值一旦设置好便不能再被更改
  • var 用来声明变量,变量的值可以在将来设置为不同的值。

在创建类或结构体的实例时,必须为所有的存储属性设置一个初始值??梢栽诔跏蓟骼镂娲⑹粜陨柚靡桓龀跏贾担梢苑峙湟桓瞿系氖粜灾底魑ㄒ宓囊徊糠?。如:

struct People {
    var age = 12
    let name = "小明"

}

class Person {
    var age: Int
    let name: String

    init(_ age: Int, name: String) {
        self.age = age
        self.name = name
    }
}

let point = People()
let person = Person(18, name: "小明")

接下来声明以下两个变量来进行查看

var age = 18
let age1 = 20

1.1 汇编分析 let 和 var


结论:可以发现两者都是一样的,直接把值复制到寄存器中。

1.2 lldb分析 let 和 var


结论:可以发现两者存储的地址是连续的,而且都在__DATA.__common这个全局区内。

1.3 SIL分析 let 和 var


从SIL文件中可以发现两者都是存储属性,都有初始值,唯一的不同就死age有set方法,age1没有。

结论:var 修饰的属性有 get 和 set 方法,而let 修饰的属性只有 get 方法,这就是 let 修饰的属性不能修改的原因。

2、计算属性

计算属性注意事项:

  • 除了存储属性,类、结构体和枚举也能够定义计算属性,计算属性并不存储值
  • 他们提供 getter 和 setter 来修改和获取值。
  • 如果只提供 getter 方法的计算属性叫做只读计算属性
  • 对于存储属性来说可以是常量或变量,但计算属性必须定义为变量。
  • 书写计算属性时候必须包含类型,因为编译器需要知道期望返回值是什么。

2.1 SIL探索

struct Square{
    //实例当中占据内存
    var width: Double
    
    //隐藏set方法,struct外部只读
    private(set) var height : Double
    
    //本质是方法,不占据内存
    var area: Double{
        get{
            return width * height
        }set{
            //系统默认新参数newValue,可通过传参的模式更改参数名
            self.width = newValue
        }
    }
}

var s = Square(width: 10,height: 10)
s.area = 30

从SIL文件中可以发现height拥有@_hasStorage标记,本质仍然是存储属性,而area属性没有。

结论:计算属性的本质就是get和set方法。

3、属性观察者

属性观察者会观察用来观察属性值的变化

  • willSet 当属性将被改变调用,即使这个值与原有的值相同
  • didSet 在属性已经改变之后调用
  • 在初始化期间设置属性时不会调用 willSet 和 didSet 观察者,只有在为完全初始化的实例分配新值时才会调用
  • 属性观察者只是对存储属性起作用
  • 当有属性观察者有继承时,调用顺序为:
    override willSet -> willSet -> 赋值 -> didSet -> override didSet

3.1 SIL探索

class SubjectName {
    var subjectName: String = ""{
        //在初始化期间设置属性时不会调用
        willSet{
            print("subjectName will set value \(newValue)")
        }
        didSet{
            print("subjectName has been changed \(oldValue)")
        }
    }
    
    init(_ subjectName:String){
        self.subjectName = subjectName
    }
}

print("begin")
let s = SubjectName("swift")
print("middle")
s.subjectName = "Swift"
print("end")

//打印结果
//begin
//middle
//subjectName will set value Swift
//subjectName has been changed swift
//end

从SIL文件中可以发现:

  • 在调用subjectName的 setter 时候,赋值之前会先调用 willSet 赋值完成之后会调用 didSet
  • 在SubjectName 的初始化函数中,subjectName是取地址直接赋值而不是调用其自身的 setter 方法

4、延迟存储属性

  • 延迟存储属性的初始值在其第一次使用时才进行计算。
  • 用关键字 lazy 来标识一个延迟存储属性。
  • lazy 属性必须是 var,不能是 let,因为 let 必须在实例的初始化方法完成之前就拥有值。
  • lazy无法保证线程安全,多线程下。
  • 当结构体包含一个延迟存储属性时,只有 var 实例变量才能访问延迟存储属性,因为延迟属性初始化时需要改变结构体的内存。

4.1 SIL探索

class Subject{
    lazy var age : Int = 18
}

var subject = Subject()


从SIL中可以发现

  • 存储属性在添加了 lazy 修饰后,该属性拥有 final 修饰符,说明 lazy 修饰的属性不能被重写。并且,它是一个可选项,意味着这个值可以是Optional.none,也就是nil。

5、类型属性

  • 类型属性其实就是一个全局变量
  • 类型属性只会被初始化一次

5.1 SIL探索

class Teacher {
    // 只被初始化一次
    static var age: Int = 18
}
Teacher.age =  20


从SIL文件可以发现属性前用了static修饰,同时生成了两个全局变量tokenage,也就说类型属性其实就是一个全局变量

从main函数中可以发现访问age变量是通过Teacher.age.unsafeMutableAddressor函数来访问,函数名为s4main7TeacherC3ageSivau,定位到该函数

通过这个函数,可以发现整个过程其实就是就是获取了token和age两个全局变量地址转化后返回。
其中:
age创建函数s4main7TeacherC3age_WZ


builtin "once" 实际上是调用了GCD中的dispatch_once_f,因此保证了只会被初始化一次。
在SIL文件中无法找到说明,直接转化成IR文件可以发现函数s4main7TeacherC3age_Wz的调用是swift_once

在swift源码中Once.h中,可以发现

所以可以得出结论builtin "once" 实际上是调用了GCD中的dispatch_once_f,因此保证了只会被初始化一次

5.2 单例

class Teacher {
    static let sharedInstance = Teacher()
    // 指定初始化器私有化,外界访问不到
    private init(){}
}
Teacher.sharedInstance

6、属性在MachO文件的位置

6.1 源码探索

swift 类的本质是HeapObject,他有两个成员变量 MetadataRefcount,其中Metadata中存放了 Description,Swift 类的属性就存放在Description的 fieldDescriptor

struct Metadata{ 
    var kind: Int
    var superClass: Any.Type
    var cacheData: (Int, Int)
    var data: Int
    var classFlags: Int32
    var instanceAddressPoint: UInt32 var instanceSize: UInt32
    var instanceAlignmentMask: UInt16
    var reserved: UInt16
    var classSize: UInt32
    var classAddressPoint: UInt32
    var typeDescriptor:TargetClassDescriptor 
    var iVarDestroyer: UnsafeRawPointer
}

class TargetClassDescriptor {
    var flags: UInt32
    var parent: UInt32
    var name: Int32
    var accessFunctionPointer: Int32
    var fieldDescriptor: Int32
    var superClassType: Int32
    var metadataNegativeSizeInWords: UInt32
    var metadataPositiveSizeInWords: UInt32
    var numImmediateMembers: UInt32
    var numFields: UInt32
    var fieldOffsetVectorOffset: UInt32
    var Offset: UInt32
    // var size: UInt32
    //V-Table
}

在源码中 TargetClassDescriptor是继承自 TargetTypeContextDescriptor ,在这个类中发现了FieldDescriptor




可以推断出FieldDescriptor 的结构体大致如下

class FieldDescriptor {
    MangledTypeName int32
    Superclass int32
    Kind uint16
    FieldRecordSize uint16
    NumFields uint32
    FieldRecords [FieldRecord]
}

其中 NumFields 代表当前有多少个属性, FieldRecords 记录了每个属性的信息,对于FieldRecords源码中其实就是FieldRecordIterator


通过源码不难发现FieldRecordIterator就是个迭代器,其中存放着FieldRecord类型的变量,

由源码不难推断出FieldRecord的数据结构为

struct FieldRecord{
    Flags uint32 
    MangledTypeName int32 
    FieldName int32
}

6.2 Mach-O源码探索

class Person {
    var age: Int = 18
    var name : String = "小明"
}

计算 ClassDescriptor 在Mach-O 中的偏移地址,然后减去虚拟基地址,找到 ClassDescriptor 的偏移

0x3F00 + 0xFFFFFF40 - 0x100000000 = 0x3E40

FieldDescriptor 在 ClassDescriptor 中是第五个属性,所以需要向后偏移 16 字节也就是3E50的位置

再次读取四个字节就是 fieldDescriptor的偏移信息,所以 fieldDescriptor 在Mach-O的位置为

0x3E50 + 0x88 = 3ED8

此时要找到FieldRecords的信息,根据结构体信息可知需要偏移16个字节,即3EE8开始分别是flag、MangledTypeName、FieldName,其中FieldName存的是偏移信息


那么可以计算出FieldName的在Mach-O的位置是

0x3EE8 + 8 + 0xFFFFFFDD - 0x100000000 = 0x3ECD

可以发现 0x3ECD 在 Mach-O 文件 reflstr 的位置,存储的是属性名称。

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

推荐阅读更多精彩内容

  • 1. 存储属性 一个存储属性就是存储在特定类或结构体实例里的一个常量或变量。 可以是变量存储属性(用关键字 var...
    DevXue阅读 210评论 0 0
  • 存储属性 在其最简单的形式中,存储属性是作为特定类或结构的实例的一部分存储的常量或变量。 存储的属性可以是变量存储...
    Joker_King阅读 377评论 0 0
  • 属性 存储属性(Stored Property)类似于成员变量这个概念存储在实例的内存中结构体,类可以定义存储属性...
    lieon阅读 476评论 0 1
  • /*储存属性储存常量或变量的要么给他默认值,要么在构造方法里初始化//计算属性计算属性不可直接储存值访问的时候调用...
    我路遇你阅读 145评论 0 0
  • 存储属性会占用实例变量的内存空间,且根据let/var关键字来生成对应的get/set方法 计算属性不会占用内存空...
    GitArtOS阅读 372评论 0 6