上一篇文章我们讲到 alloc 在开辟内存空间之前,对对要分配的内存空间提前进行计算,并最终使用 16 字节对其方法进行对其,提升了读取的效率。但是16 字节对其之前,如何计算对象实际需要的空间呢?
1. 对象内存分析
先展示一段测试代码
1 | @interface LGPerson : NSObject |
这里可以发现,对于类本身,sizeof 可以查询到它所占用的地址空间只有 8 位。
而 class_getInstanceSize 为何是 40?没有 16 进制对其?
其实 class_getInstanceSize 获取到的是实例对象实际需要占用的内存空间,而且实际使用一般只会做 8 字节对齐。
而 malloc_size 则是系统实际为该实例对象分配的空间。也就是经过 16 字节对其后的效果。
2. 结构体内存对齐
看完了对象,我们看下结构体是如何进行内存对齐的。(实际上要比对象对其要简单些,并且不涉及 16 进制对其的优化,能更直观的得出内存空间实际占用的计算)
预备知识
在展示实例之前,我们先看下实例中用到的类型的内存空间情况
1 | NSLog(@"double size %lu", sizeof(double)); |
初级案例
ok,这些类型的空间大小大家应该熟悉了。下面介绍实例
1 | typedef struct Str1 { |
我们先来看下结构体 Str1 的内存大小 - 16。
实际上,基础类型的存储位置为栈内存,而结构体的存储也使用了内存对其的技术,用于减少 cpu 的访问内存次数,提升读取效率。
首先我们需要了解一下结构体内存对其的规则:
- 结构体变量的首地址是其最长基本类型成员的整数倍;
- 结构体每个成员相对于结构体首地址的偏移量(offset)都是成员大小的整数倍,如不满足,对前一个成员填充字节以满足;
- 结构体的总大小为结构体最大基本类型成员变量大小的整数倍;
- 结构体中的成员变量都是分配在连续的内存空间中。
1 | typedef struct Str1 { |
所以,根据规则,我们得出上述结构体 Str1 内存大小为 24 字节。
进阶案例
上面案例中,str2 是一个嵌套类型结构体,他的大小又是如何求得的呢?
💡知识点:结构体嵌套的内存对其方式
如果一个结构体B里嵌套另一个结构体A,还是以最大成员类型的字节对齐,但是结构体A存储起点为A内部最大成员整数倍的地方。(struct B里存有struct A,A里有char,int,double等成员,那A应该从8的整数倍开始存储。),结构体A中的成员的对齐规则仍满足自身的规则
1 | typedef struct Str2 { |