IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 移动开发 -> Swift类型属性底层研究 -> 正文阅读

[移动开发]Swift类型属性底层研究

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

基本语法

  • 类型属性的语法和成员属性类似的地方包括:可以定义存储属性和计算属性,也可以添加存储属性监听器
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进行, 确保只会初始化一次,并且是线程安全的
  • 编译器对类型属性和全局变量添加了内存安全的控制,避免了访问的读写冲突,使代码更加安全
  移动开发 最新文章
Vue3装载axios和element-ui
android adb cmd
【xcode】Xcode常用快捷键与技巧
Android开发中的线程池使用
Java 和 Android 的 Base64
Android 测试文字编码格式
微信小程序支付
安卓权限记录
知乎之自动养号
【Android Jetpack】DataStore
上一篇文章      下一篇文章      查看所有文章
加:2022-09-04 01:23:00  更:2022-09-04 01:24:23 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 4:46:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码