swift -(5) 汇编分析结构体、类的内存布局

一、结构体

在 Swift 标准库中,绝大多数的公开类型都是结构体,而枚举和类只占很小一部分

比如Bool Int Double String Array Dictionary等常见类型都是结构体

①  struct Date {
②         var year: Int
③         var month: Int
④         var day: Int ⑤  }
⑥  var date = Date(year: 2019, month: 6, day: 23)

 所有的结构体都有一个编译器自动生成的初始化器( initializer ,初始化方法、构造器、构造方法)

在第⑥行调用的,可以传入所有成员值,用以初始化所有成员(存储属性,Stored Property

二、结构体的初始化器

编译器会根据情况,可能会为结构体生成多个初始化器,宗旨是:保证所有成员都有初始值

从这列子中可以看出,编译器最终选择什么初始化器和结构体原始定义有没有赋默认值有关,编译器选择的初始化器会保证所有成员都有初始值;

三、这能编译通过么

struct Point {  var x: Int? var y: Int?
}
var p1 = Point(x: 10, y: 10)
var p2 = Point(y: 10)
var p3 = Point(x: 10)
var p4 = Point()

根据编译器初使化饿的本质保证所有成员都有初始值

可选项都有个默认值nil

因此可以编译通过

四、自定义初始化器

注: 一旦在定义结构体时自定义了初始化器,编译器就不会再帮它自动生成其他初始化器

因此下边的代码后边三句会拨错

五、窥探初始化器的本质分析自定义初始化器和编译器会我们生成的初始化器是一样的么

第一句 编译器的初始化qi
struct Point {
var x: Int = 0 var
y: Int = 0
}
var p = Point()

第二句  自己的初始化qi
struct Point { 
var x: Int 
var y: Int
init() {  
x = 0 
y = 0
   }
}
var p = Point()


通过上面的汇编指令可以看出 编译器自己调用的初始化qi和我们自己写的初始化qi其实是一样的

六、结构体占用内存大小

七、

类的定义和结构体类似,但编译器并没有为类自动生成可以传入成员值的初始化器

如何里面的x和y 没有初使值,你调用无参会报错,因为你的对象创建完成后,你里面的x和y成员没有值是不安全的

如果类的所有成员都在定义的时候指定了初始值,编译器会为类生成无参的初始化器;

成员的初始化是在这个初始化器中完成的

八、结构体与类的本质区别

结构体是值类型(枚举也是值类型),类是引用类型(指针类型)

九、值类型 的深拷贝

值类型赋值给var let或者给函数传参,是直接将所有内容拷贝一份

类似于对文件进行copy paste操作,产生了全新的文件副本。属于深拷贝(deep copy 

p2.x = 11 p2.y = 22
// 请问p1.x和p1.y是多少?

依然是原来的10和20

十 、值类型的赋值操作

在Swift标准库中,为了提升性能,String Array Dictionary Set采取了Copy On Write的技术比如仅当有“写”操作时,才会真正执行拷贝操作

对于标准库值类型的赋值操作, Swift 能确保最佳性能,所有没必要为了保证最佳性能来避免赋值

 建议 :不需要修改的,尽量定义成let

为了提升性能,String Array Dictionary Set采取了Copy On Write的

var s1 = "Jack" var s2 = s1

如有了这句才会有深拷贝,在之前没有发生写操作,直接就是浅拷贝

s2.append("_Rose")

print(s1) // Jack

print(s2) // Jack_Rose

十一、引用类型的赋值

s2.width = 11  s2.height = 22

// 请问s1.widths1.height是多少 11 22 

十二、值类型、引用类型的let

十三、嵌套类型

十三、堆空间内存占用情况

class Point  {
var x = 11
var test = true 
var y = 22
}
var p = Point()
class_getInstanceSize(type(of: p)) // 40
class_getInstanceSize(Point.self) // 40

内存地址的前16字节存(指向类型信息和引用计数),堆空间会检查是不是16的倍数,堆空间内存对其是48
根据内存对其最大的倍数,这里是8,所以这里分配的是40个字节,内存地址按8对其,实际用到的是16+8+8+3 =33

总结-汇编内存存值常用参数总结   

1、 1.56

rax、rdx常作为函数返回值使用
rdi、rsi、rdx、rcx、r8、r9等寄存器常用于存放函数参数

 rsp、rbp用于栈操作

rip作为指令指针
 -存储着CPU下一条要执行的指令的地址
 -一旦CPU读取一条指令,rip会自动指向下一条指令(存储下一条指令的地址)

2、1.30

内存地址格式为:0x4bdc(%rip) ,一般是全局变量 ,全局区(数据段)

内存地址格式为:-0x78(%rbp) ,一般是局部变量 ,栈空间

内存地址格式为:0x10(%rax) ,一般是堆空间 2.07

0x1000047b8 <+144>: str    x0, [sp, #0x8]

    0x1000047bc <+148>: str    x1, [sp]

->  0x1000047c0 <+152>: adrp   x0, 3

    0x1000047c4 <+156>: add    x0, x0, #0x9a0            ; "b "

    0x1000047c8 <+160>: mov    w8, #0x2                  ; =2 

    0x1000047cc <+164>: mov    x1, x8

<+164> 前面所有的占用多少字节

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值