美文网首页
Swift类型属性底层研究

Swift类型属性底层研究

作者: chonglingliu | 来源:发表于2022-08-29 14:58 被阅读0次

我们研究过成员属性的一些具体实现细节,本文我们来研究下类型属性的底层逻辑。

基本语法

  • 类型属性的语法和成员属性类似的地方包括:可以定义存储属性和计算属性,也可以添加存储属性监听器
struct Sequence {
    static var first: Int = 1 // 存储属性
    static var second: Int {  // 计算属性
        get {
            return first
        }
        set {
            first = newValue
        }
    }
    static var third: Int = 3 { // 存储添加属性监听器
        didSet {
            print("third didSet")
        }
        willSet {
            print("third willSet")
        }
    }
}

区别是类型属性要用static进行修饰

  • 类型属性不能用lazy修饰,因为类型属性默认就是already-lazy global
不能用lazy修饰

swift_once

实现分析

类型属性默认是懒加载,我们来看看底层的实现逻辑。

<!-- 测试代码 -->
struct Sequence {
    static var first: Int = 1
}

func test() {
    Sequence.first = 2
}

test()
  • 获取Sequence.first的内存地址:Sequence.first.unsafeMutableAddressor
Sequence.first.unsafeMutableAddressor
  • 如果地址不存在,利用swift_once进行变量的初始化
swift_once
  • swift_once底层调用的是dispatch_once_f
dispatch_once_f

我们得知:编译器会封装一个初始化函数,作为dispatch_once_ffn参数进行初始化调用

  • fn函数封装
// one-time initialization function for first
sil private [global_init_once_fn] @$s4main8SequenceV5first_WZ : $@convention(c) () -> () {
bb0:
  alloc_global @$s4main8SequenceV5firstSivpZ      // id: %0
  %1 = global_addr @$s4main8SequenceV5firstSivpZ : $*Int // user: %4
  %2 = integer_literal $Builtin.Int64, 1          // user: %3
  %3 = struct $Int (%2 : $Builtin.Int64)          // user: %4
  store %3 to %1 : $*Int                          // id: %4
  %5 = tuple ()                                   // user: %6
  return %5 : $()                                 // id: %6
} 

通过SIL分析,我们得知:编译器会封装一个初始化函数,大体的实现逻辑是:

  • 得到变量的内存地址, 类似于:var ptr = withUnsafePointer(to: &Sequence.first) { $0 }
  • 将1赋值给这个内存地址上,类似于:ptr.pointee = 2
总结
  • 编译器会将var first: Int = 1封装成一个函数,函数体是先获取变量指针,然后给指针所指的内存地址赋值为初始值
  • 类型属性底层是通过dispatch_once_f进行初始化,确保只会初始化一次,并且是线程安全的

全局变量

从图一的编译器错误提示我们可以得知,类型属性本质就是全局变量,只是有访问权限限定。

let zero: Int = 0

struct Sequence {
    static var first: Int = 1
}

我们利用实例代码进行分析。

SIL分析
@_hasStorage @_hasInitialValue var zero: Int { get set }

struct Sequence {
  @_hasStorage @_hasInitialValue static var first: Int { get set }
  init()
}

// zero
sil_global hidden @$s4main4zeroSivp : $Int

// static Sequence.first
sil_global hidden @$s4main8SequenceV5firstSivpZ : $Int

我们看到SIL语法中,全局变量和类型属性的定义是完全相同的。

内存验证
func test() {
    let ptr1 = withUnsafePointer(to: &zero) { UnsafeRawPointer($0) }
    let ptr2 = withUnsafePointer(to: &Sequence.first) { UnsafeRawPointer($0) }
    print("\(ptr1) \(ptr2)")
}
// 0x100008000 0x100008008

通过查看内存地址,我们得到的结果是zeroSequence.first的内存地址是连续挨在一起的。zero肯定是全局变量,所以Sequence.first本质上也是一个全局变量。

全局变量的更多用法

既然类型属性是全局变量,那全局变量应该也可以是计算属性等。其实确实也是可以这样写的:

var zero: Int = 0
var one: Int {
    get {
        zero
    }
    set {
        zero = newValue
    }
}
var two: Int = 2 {
    willSet {
        
    }
    didSet {
        
    }
}

全局变量的语法和类型属性的语法也是一致的。

变量内存安全(参考地址

前面我们看到了类型属性本质是通过swift_once得到了变量内存地址指针。Swift编译器可以(也仅仅只有编译器可以)获取到全局变量的内存地址指针。

为什么需要获取变量的内存地址指针呢?这涉及到内存安全的部分

Swift会保证同时访问同一块内存时不会冲突,通过约束代码里对于存储地址的写操作,去获取那一块内存的访问独占权。避免了读写冲突。

变量内存安全是通过swift_beginAccessswift_endAccess等方法类控制的。

变量内存安全
swift_beginAccess
swift_beginAccess AccessSet::insert

逻辑总结:

  1. 先将内存指针封装成Access对象
  2. Access对象的封装的内存指针如果在SwiftTLSContext::get().accessSet数组中不存在,说明目前没有其他方法占用该内存地址,可以访问,并且将Access对象保存起来;
  3. Access对象的封装的内存指针如果在SwiftTLSContext::get().accessSet数组中存在,说明该内存地址已经有访问存在了,如果所有的访问都是读访问,则不认为是冲突,可以继续访问,否则就会报访问冲突错误。
swift_endAccess
swift_endAccess

逻辑总结:
将当前的访问从SwiftTLSContext::get().accessSet数组中移除,也就是将本次内存访问移除。

总结

  • 类型属性本质上是全局变量,只是访问权限有所限制
  • 类型属性和全局变量可以是存储属性,计算属性,也可以添加属性监听器,但是不能添加懒加载的lazy关键字
  • 类型属性是懒加载的,通过dispatch_once_f进行, 确保只会初始化一次,并且是线程安全的
  • 编译器对类型属性和全局变量添加了内存安全的控制,避免了访问的读写冲突,使代码更加安全

相关文章

  • Swift类型属性底层研究

    我们研究过成员属性的一些具体实现细节,本文我们来研究下类型属性的底层逻辑。 基本语法 类型属性的语法和成员属性类似...

  • Swift笔记(一)属性、析构、调用OC单例

    目录 swift属性存储属性懒加载属性计算属性属性监听类型属性 swift析构函数 swift调用OC单例类方法 ...

  • Swift 属性

    Swift 属性 在Swift中属性主要分为存储属性、计算属性、延迟存储属性、类型属性这四种,并且Swift还提供...

  • Swift进阶(三)--- 属性

    Swift的属性 在swift中,属性主要分为以下几种: 存储属性 计算属性 延迟存储属性 类型属性 一:存储属性...

  • Swift 可选类型Optional

    Swift 可选类型Optional [TOC] 前言 本将以Swift中的可选类型为入口,介绍: 可选类型的底层...

  • Swift底层探索(二):类型属性、值类型&引用类型

    延迟存储属性 延迟属性赋值&大小 在hotpot.age赋值前和赋值后各打个断点,x/8g看下内存地址,可以看到赋...

  • 每天学一点Swift----面向对象上(十一)

    十三.类型属性和类型方法 1.通过前面的学习,已经知道Swift的类型中有5种成员:属性(存储属性和计算属性)、方...

  • Swift 学习笔记(三)

    Swift属性 Swift属性将值跟特定的类,结构体,枚举关联。分为存储属性和计算属性,通常用于特定类型的实例。属...

  • Swift 类型属性、类型方法

    简述 Swift中的类型(class、struct、enum等)属性和类型方法分别属于静态属性和静态方法。这种类型...

  • Swift 静态属性

    Swift 属性按照 定义方式 分为: 存储属性计算属性 按照 调用方式 分为: 实例属性类型属性(静态属性) 实...

网友评论

      本文标题:Swift类型属性底层研究

      本文链接:https://www.haomeiwen.com/subject/dfzhnrtx.html