美文网首页iOS底层原理
iOS开发之内存对齐

iOS开发之内存对齐

作者: 爱看书de图图 | 来源:发表于2020-09-10 17:29 被阅读0次

  我们先来看两个结构体:

#import <Foundation/Foundation.h>
#import <objc/runtime.h>
#import <malloc/malloc.h>

struct HKStruct1 {
    double a;   // 8
    char b;     // 1
    int c;      // 4
    short d;    // 2
}struct1;
struct HKStruct2 {
    double a;   //8
    int b;      //4
    char c;     //1
    short d;    //2
}struct2;
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        NSLog(@"%lu-%lu",sizeof(struct1),sizeof(struct2));
    }
    return 0;
}


  在arm64架构下,int占用4个字节,char占用1个字节,double8个字节,short2个字节,那么结构体struct1struct2应该占用的大小是15个字节,即大小为15。但是我们调用函数sizeof后的结果却为24-16(如下图),这就是系统内存对齐后的结果。
内存对齐的概念

  就是编译器为程序中的每个“数据单元”安排在适当的位置上,如果不对齐,那么处理器访问这片内存就需要两次访问,而访问对齐后的内存只需要一次。即空间换时间

内存对齐的原则

1、数据成员对⻬规则:结构(struct)(或联合(union))的数据成员,第⼀个数据成员放在offset为0的地⽅,以后每个数据成员存储的起始位置要从该成员⼤⼩或者成员的⼦成员⼤⼩(只要该成员有⼦成员,⽐如说是数组,结构体等)的整数倍开始(⽐如int为4字节,则要从4的整数倍地址开始存储。
2、结构体作为成员:如果⼀个结构⾥有某些结构体成员,则结构体成员要从其内部最⼤元素⼤⼩的整数倍地址开始存储.(struct a⾥存有struct b,b⾥有char,int ,double等元素,那b应该从8的整数倍开始存储)。
3、结构体的总⼤⼩,也就是sizeof的结果,必须是其内部最⼤成员的整数倍,不⾜的要补⻬。
4、每个特定平台上的[编译器]都有自己的默认“对齐系数”(也叫对齐模数)。程序员可以通过[预编译]命令#pragma pack(n),n=1,2,4,8,16来改变这一系数,其中的n就是你要指定的“对齐系数”。
接下来我们详细解析,为什么两个结构体的内存大小不一致:

结构体1
struct HKStruct1 {
    double a;   // 长度8 < 16 按8对齐;起始offset=0 0%4=0;存放位置区间[0,7] 
    char b;     // char型,长度1 < 8 按1对齐;起始offset=8 8%1=0;存放位置区间[4] 
    int c;      // 4  int型,长度4 < 16 按4对齐;起始offset=9 `9%4!=0`,12%4=0;存放位置区间[12,15]
    short d;    // 2  char型,长度2 < 16 按2对齐;起始offset=16 16%2=0;存放位置区间[16,17]
}struct1;
  • double类型,长度8个字节,起始位置offset = 0,0%4 = 0;存放位置区间[0,7]
  • char类型,长度1个字节,起始位置offset = 8 ,8%1 = 0,存放位置区间[8]
  • int类型,长度4个字节,起始位置offset = 9,9%4 != 0,因为不能整除,所以往后移,到12的位置,即12%4 = 0;存放位置区间[12,15]
  • short类型,长度2个字节,起始位置offset = 16,16%2 = 0,存放位置区间[16,17]
  • 所以结构体的大小为24,即8的整数倍。
结构体2
struct HKStruct2 {
    double a;   //8
    int b;      //4
    char c;     //1
    short d;    //2
}struct2;
  • double类型,长度8个字节,起始位置offset = 0,0%4 = 0;存放位置区间[0,7]
  • int类型,长度4个字节,起始位置offset = 8 ,8%4 = 0,存放位置区间[8,11]
  • char类型,长度1个字节,起始位置offset = 12,12%4 = 0,存放位置区间[12,13]
  • short类型,长度2个字节,起始位置offset = 14,14%2 = 0,存放位置区间[14,15]
  • 所以结构体的大小为16,即8的整数倍。
实例分析
结构体嵌套结构体
//1、结构体嵌套结构体
struct Mystruct3{
    double b;   //8
    int c;      //4
    short d;    //2
    char a;     //1
    struct HKStruct2 str;//16
}struct3;
实例2结构体嵌套结构体
struct Mystruct4{
    short d;    //2字节
    char a;     //1字节
    struct Test {
        double x;
    }test;
}struct4;

打印的结果和我们上面分析的结果相同。

相关文章

  • iOS内存对齐

    这篇文章我们来探索一下iOS内存对齐的原理,在探索完内存对齐原理之后,你就会明白内存对齐的好处。 在讲述内存对齐时...

  • iOS底层探究 - 内存对齐

    目录1:内存对齐的原因2:内存对齐的规则3:结构体内存分配演练以及在iOS中对象成员的内存分配探索 一 :内存对齐...

  • iOS开发 内存对齐原理

    1.文字描述 数据成员对⻬规则:结构(struct)(或联合(union))的数据成员,第一个数据成员放在offs...

  • iOS开发之内存对齐

      我们先来看两个结构体:   在 架构下, 占用4个字节, 占用1个字节, 8个字节, 2个字节,那么结构体 和...

  • iOS 开发 内存对齐(练习)

    目录 内存对齐规则 对齐系数 面试题演练 一、内存对齐规则 (关于面试题中结构体内存对齐计算总结) 1.1、数据成...

  • iOS Struct嵌套类型的内存分析

    一,为什么要内存对齐 在iOS开发过程中,甚至任何一门开发语言,对于内存的资源都是极其宝贵的,不能随意的浪费,所以...

  • iOS底层之内存对齐算法解析

    目前但凡一个iOS岗面试都会问个内存对齐问题,那么什么是字节对齐?成员变量对齐和对象内存对齐有什么区别?今天我来为...

  • iOS底层之 内存对齐

    数据类型大小 首先附上C和OC,各数据类型的大小表。 COC32位64位boolBOOL(64位)11signed...

  • iOS之底层内存对齐

    引言 内存对齐是内存里面一个很重要的词汇,可是大部分开发者对这个词汇的含义都是一知半解。 输出结果: 结果分析: ...

  • iOS 内存对齐

    在上一边文章中,我们在简单介绍内存对齐,今天我们更加深入一点: 一、获取内存大小的三种方式 先看下面这段代码: 看...

网友评论

    本文标题:iOS开发之内存对齐

    本文链接:https://www.haomeiwen.com/subject/gqypektx.html