首页 > 系统相关 >Swift开发基础07-内存布局

Swift开发基础07-内存布局

时间:2024-07-16 10:19:59浏览次数:15  
标签:07 计数 ARC 引用 print Swift 内存

了解Swift的内存布局和底层原理对于编写高性能和内存高效的应用非常重要。接下来,我将更详细地介绍Swift的内存管理机制和一些底层实现细节,包括内存布局、ARC(自动引用计数)、引用类型和值类型的区别,及其在底层的实现。

内存布局(Memory Layout)

栈(Stack)

栈内存用于存储函数调用帧(Call Stack)、局部变量和函数参数。栈的特点是LIFO(Last In, First Out),即最后进入的最先出去。栈内存分配和释放速度非常快,但栈的空间是有限的。

堆(Heap)

堆内存用于存储动态分配的内存,比如对象、闭包等。堆的内存管理相对复杂,需要显式管理分配和释放。Swift通过ARC来自动管理堆内存。

值类型与引用类型的内存布局

  • 值类型(Value Type):
    • 存储在栈上或嵌入到引用类型中的对象内部。
    • 包括基本类型(整数、浮点数、布尔值)、结构体、枚举等。
    • 每次赋值会拷贝一份独立的数据。
  • 引用类型(Reference Type):
    • 存储在堆上,并通过指针引用管理。
    • 包括类(Class)、闭包(Closure)等。
    • 赋值操作只会复制指针,而不会复制对象本身。

自动引用计数(ARC)

ARC在运行时管理引用类型实例的内存生命周期。每个引用类型实例有一个引用计数(Reference Count),当引用计数为零时,ARC会自动释放该实例的内存。

ARC的底层机制

  1. 引用计数增加(retain):

    • 每当有新的引用指向一个对象,其引用计数加一。
    • 底层实现为调用objc_retain
  2. 引用计数减少(release):

    • 每当一个引用不再指向该对象,其引用计数减一。
    • 底层实现为调用objc_release
  3. 内存释放:

    • 当引用计数为零时,对象的deinit方法被调用,然后释放对象占用的内存。

ARC示例

class ExampleClass {
    let name: String

    init(name: String) {
        self.name = name
        print("\(name) is initialized")
    }
    
    deinit {
        print("\(name) is being deallocated")
    }
}

var object1: ExampleClass? = ExampleClass(name: "Object1")  // 引用计数+1
var object2 = object1  // 引用计数+1,此时引用计数为2
object1 = nil          // 引用计数-1,此时引用计数为1
object2 = nil          // 引用计数-1,此时引用计数为0,对象被释放

 

上下文捕获(Context Capture)

捕获机制

闭包可以捕获并存储其上下文中的变量和常量,即使这些变量和常量在闭包创建时已经作用域结束。Swift通过将被捕获的变量存储在堆上来实现这种捕获。

func makeIncrementer(incrementAmount: Int) -> () -> Int {
    var total = 0
    let incrementer: () -> Int = {
        total += incrementAmount
        return total
    }
    return incrementer
}

let incrementByTwo = makeIncrementer(incrementAmount: 2)

print(incrementByTwo()) // 输出: 2
print(incrementByTwo()) // 输出: 4

 

在这个例子中,totalincrementAmount被捕获到堆中,并在闭包中引用。

Swift底层对象模型

Swift类的内存布局

Swift类的内存布局包含一个头部和实例数据部分。头部通常包含:

  • 引用计数:用于ARC。
  • 类元数据指针:指向类型元数据(Class Metadata)。

方法派发(Method Dispatch)

Swift支持多种方法派发方式:

  1. Direct Dispatch(直接派发):主要用于值类型或没有多态性的类。
    • 编译时决定,性能最高。
  2. VTable Dispatch(虚表派发):用于引用类型的多态方法调用。
    • 通过虚表(VTable)找到方法的具体实现。
  3. Message Dispatch(消息派发):主要用于兼容Objective-C的类。
    • 通过运行时的消息机制进行方法调用。
class BaseClass {
    func printMessage() {
        print("BaseClass")
    }
}

class SubClass: BaseClass {
    override func printMessage() {
        print("SubClass")
    }
}

let instance: BaseClass = SubClass()
instance.printMessage() // 运行时通过VTable找到SubClass的方法,输出: "SubClass"

 

内存安全(Memory Safety)

  1. 变量初始化:

    • Swift强制在使用变量之前进行初始化,避免未初始化变量的使用。
  2. 数组越界检查:

    • Swift在访问数组元素时,会进行边界检查,防止越界访问。
  3. 原子性:

    • Swift对内存的读写操作是原子的,可以避免数据竞争。

组合的安全示例

综合上述内容,以下是一个更复杂的示例,展示了闭包捕获、ARC、VTable派发以及内存安全性:

 
class Counter {
    var count = 0
    
    func increment() -> Int {
        count += 1
        return count
    }
    
    deinit {
        print("Counter is being deallocated")
    }
}

func makeCounter() -> () -> Int {
    let counter = Counter()
    
    let increment: () -> Int = { [weak counter] in
        guard let counter = counter else { return 0 }
        return counter.increment()
    }
    
    return increment
}

let incrementer = makeCounter()
print(incrementer()) // 输出: 1
print(incrementer()) // 输出: 2

 

上面的示例展示了如何在Swift中效率和安全地处理内存。通过理解这些底层细节,能更好地编写性能高效、内存安全的Swift代码。

标签:07,计数,ARC,引用,print,Swift,内存
From: https://www.cnblogs.com/chglog/p/18047564

相关文章

  • 题解:P10724 [GESP202406 七级] 区间乘积
    思路看到\(a_i\)很小,不难想到状压一类的东西。考虑把每个数的质因数当做二进制位,这个二进制位的\(1/0\)代表含有这个质因数的奇偶,再做一个异或前缀和,显然完全平方数的质因子个数一定为偶数,根据异或的性质,两个相同的数异或才为\(0\)所以只需要找到异或前缀和中相同的数的个......
  • C语言 ——— 浮点数类型 在 内存中 的 存储模式
    浮点数存储规则根据国际标准IEEE754(电气和电子工程协会)规定:任意一个浮点数F的二进制都可以表示成以下形式:科学计数法(-1)^S*M*2^E解析科学计数法: 1.解析:(-1)^S(-1)^S 表示的是浮点数F的符号位当S=0时,原式=(-1)^0= 1,此时的 1就表示浮点数F为......
  • Spark _Exam_ 20240715
    SparkExam20240715ConclusionSB出题人出DP场,T1靠小常数通过不给提示干死选手,T2出题人认为思维难度低代码5KB,NOIP场的T3放黑题,T4又是区间DP\(\mathcalO(n^6)=117649000000\)竟然能够通过?你代码常数真的小!好的喷完了。这种场的后果就是,平均分50,最高90,最低0实际上如......
  • Java 中有哪几种基本数据类型?请分别列出它们并简述每种数据类型的特点及其在内存中的
    在Java的世界里,数据是构建应用程序的基石。为了高效地处理这些数据,Java设计了一系列基础数据类型,它们直接映射到计算机硬件上,因此在性能和内存使用上更为高效。我们常说的Java八大基本数据类型,涵盖了整数、浮点数、字符和布尔值,下面我将一一介绍它们的特点以及在内存中的占用......
  • .NET周刊【7月第2期 2024-07-14】
    国内文章开源GTKSystem.Windows.Forms框架让C#winform支持跨平台运行https://www.cnblogs.com/easywebfactory/p/18289178GTKSystem.Windows.Forms框架是一种C#winform应用程序跨平台界面开发框架,兼容C#原生控件,无需额外学习,支持跨平台运行。其优势包括开源、与visualstudio......
  • 【vue组件库搭建07】Vitest单元测试
    vitest官网vue-test-utils我们的测试框架选择的是Vitest和vue-test-utils。两者的关系为:Vitest提供测试方法:断言、Mock、SpyOn等方法。vue-test-utils:挂载和渲染组件:VueTestUtils允许您在隔离中挂载组件,这意味着您可以测试单个组件而不必担心其子组件或需要......
  • AcWing 2074:倒计数 ← 双指针算法
    【题目来源】https://www.acwing.com/problem/content/2076/【题目描述】艾弗里有一个由N个正整数构成的数组。数组中的第i个整数是Ai。如果一个连续的子数组的长度为m,并且按顺序包含整数m,m−1,m−2,…,2,1,则称它为m倒计数。例如,[3,2,1]是3倒计数。请帮助艾......
  • 内存管理-19-vmlinux.lds.S分析
    基于msm-5.4一、简介链接器主要任务是将符号引用解析到符号定义上,将多个目标文件(.o)和库文件合并成为一个可执行文件或者动态链接库,生成符号表,并对程序代码做最后的检查和优化。这个链接脚本在Linux内核里就是vmlinux.lds.S文件。vmlinux.lds.S编译后会在out/target目录......
  • Memcached:高性能内存缓存系统详解及实战
    引言在高并发的Web应用中,数据库往往成为性能瓶颈。为了提高数据读取速度和减少数据库负载,引入缓存机制变得至关重要。Memcached正是这样一款高性能、分布式内存对象缓存系统,它通过在内存中缓存数据来加速动态Web应用,从而极大地改善用户体验。本文将深入探讨Memcached的工作原......
  • 0715鲜花——写在阿根廷夺冠之后
    首先声明:本人只是因为喜欢体育从而喜欢足球,但是不会踢球,不喜勿喷一.前言今天可能是个被足球所眷顾的日子欧洲杯和美洲杯除了世界杯以外最著名两个的洲际比赛的决赛发生在同一天让我先祝贺西班牙和阿根廷分别夺得欧洲杯和美洲杯的冠军!!这两个比赛,其中新创造的许多记录,我不想谈......