iOS底层探索002-内存对齐

2020-09-08  本文已影响0人  星星1024

iOS底层探索-目录

前言

计算机内存都是以字节为单位划分的,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但是实际的计算机系统对基本类型数据在内存中存放的位置有限制,它们会要求这些数据的首地址的值是某个数k(通常它为4或8的倍数,这就是所谓的内存对齐.

1. 内存对齐的原因

我们都知道内存是以字节为单位,但是大部分处理器并不是按字节块来存取内存的.它一般会以2字节,4字节,8字节,16字节甚至32字节为单位来存取内存,我们将上述这些存取单位称为内存存取粒度.

eg:以32位CPU为例,实际寻址步长为4个字节,即只对编号为4的倍数的内存寻址

内存对齐.png image.png

2. 内存对齐原则

对齐系数: 1

#pragma pack(1)
struct MStruct1 {
    char a;        //1字节
    double b;       //8字节
    int c;         //4字节
    short d;       //2字节
} MyStruct1;
#pragma pack()

设置对齐系数为1:打印结果

MyStruct1:15 - 
对齐系数-1.png

对齐系数: 2

#pragma pack(2)
struct MStruct1 {
    char a;        //1字节
    double b;       //8字节
    int c;         //4字节
    short d;       //2字节
} MyStruct1;
#pragma pack()

设置对齐系数为2:打印结果

MyStruct1:16 - 
对齐系数-2.png

对齐系数: 4

#pragma pack(4)
struct MStruct1 {
    char a;        //1字节
    double b;       //8字节
    int c;         //4字节
    short d;       //2字节
} MyStruct1;
#pragma pack()

设置对齐系数为4:打印结果

MyStruct1:20 - 
对齐系数-4.png

对齐系数: 8

对齐系数默认为成员最大元素大小

struct MStruct1 {
    char a;        //1字节
    double b;       //8字节
    int c;         //4字节
    short d;       //2字节
} MyStruct1;

struct MStruct2 {
    double b;       //8字节
    char a;        //1字节
    short d;       //2字节
    int c;         //4字节
} MyStruct2;

设置对齐系数为8:打印结果

MyStruct1:24 - 
MyStruct2:16 -
MStruct1:对齐系数-8.png
MStruct2:对齐系数-8.png

结构体嵌套

结构体嵌套:对齐系数是结构体最大成员的大小

struct MStruct1 {
    char a;        //1字节 8
    double b;       //8字节 8 (最大成员大小)
    int c;         //4字节  4
    short d;       //2字节 4
} MyStruct1;

struct MStruct2 {
    char a;        //1字节
    short d;       //2字节
    int c;         //4字节
    struct MStruct1 s1;
} MyStruct2;

打印结果:

MyStruct1:24 - 
MyStruct2:32 -

OC中类对象的内存分配

@interface Person : NSObject

@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@property (nonatomic, assign) long height;
@property (nonatomic, strong) NSString *job;

@property (nonatomic, assign) int sex;
@property (nonatomic) char ch1;
@property (nonatomic) char ch2;

@end

    NSLog(@"%lu - %lu",class_getInstanceSize([person class]),malloc_size((__bridge const void *)(person)));

打印结果:

40 - 48

通过打印发现对象本身大小和系统为对象分配的空间不一致:

下一篇: iOS底层探索003-isa分析

上一篇 下一篇

猜你喜欢

热点阅读