通过内存对齐分析IOS中的对象内存占用

总所周知,oc对象底层是由结构体实现的,所以通过分析结构体内存占用情况可以更好的理解oc对象的内存占用。

1.把OC对象编译成结构体

有如下代码:

#import <Foundation/Foundation.h>
#import "Person.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        NSLog(@"Hello, World!");
        
        Person *per = [[Person alloc] init];
        NSLog(@"per:%@",per);
        
    }
    return 0;
}

我们可以通过clang命名把.m文件编译成.cpp文件,进而可以清楚的看到为什么说oc对象底层是结构体实现。

//clang命令
clang -rewrite-objc main.m [-o 别名]

编译后如下:

struct Person_IMPL {
    struct NSObject_IMPL NSObject_IVARS;
};

2.结构体内存占用分析

2.1 常用数据类型内存占用大小

我们都知道,在不同位数的编译器环境下,数据类型不同其占用字节大小也不相同,区别如下:

  • 32位编译器


    32位编译器下.png
  • 64位编译器

    64位编译器下.png

2.2 结构体内存对齐规则

为了方便cpu更加快速地读取存放在内存中的数据,内存在存放数据时会按照一定的规则来排列,规则如下:

  • 数据成员对⻬规则:结构(struct)(或联合(union))的数据成员,第
    ⼀个数据成员放在offset为0的地⽅,以后每个数据成员存储的起始位置要从该成员⼤⼩或者成员的⼦成员⼤⼩(只要该成员有⼦成员,⽐如说是数组,结构体等)的整数倍开始(⽐如int为4字节,则要从4的整数倍地址开始存
    储。
  • 结构体作为成员:如果⼀个结构⾥有某些结构体成员,则结构体成员要从
    其内部最⼤元素⼤⼩的整数倍地址开始存储.(struct a⾥存有struct b,b⾥有char,int ,double等元素,那b应该从8的整数倍开始存储.)
  • 收尾⼯作:结构体的总⼤⼩,也就是sizeof的结果,.必须是其内部最大成员的整数倍,不⾜的要补⻬。

2.3 内存对齐计算

2.3.1 普通结构体

例如有下面这样一个结构体,其内存大小为24.分析如下:

struct AA{
    double a;   //[0,7]
    char b;     //[8]
    int c;      //[12,15]
    int d;      //[16,19]
};
printf("大小为:%d",sizeof(struct AA));//24

我们可以知道sizeof可以用来计算对象类型所占内存大小,按照规则1我们可以将结构体AA对象排列如下:


栈排列.png

我们可以看到,a是第一个成员且长度大小为8,所以占位序号为[0,7];
b的长度大小为1,所以占位序号为[8];c的长度大小为4,但是根据规则一,这里c不能从序号9开始,因为9不满足对齐数(4)的整数倍,所以要从12开始排列;同理d的占位序号为[16,19],那么整个结构体大小为19+1=20字节,又因为20不满足规则三,所以总大小应该是最大长度(8)的最小整数倍且不小于20字节的数,即24.

2.3.2 嵌套结构体

例如结构体嵌套的情况:

struct BB{
    double a;       //[0,7]
    struct AA b;    //[8,31]
    char c;         //[32]
};
printf("BB:%lu\n",sizeof(struct BB));//40

思路分析:

  • a长度为8且为首元素,所以占位序号为[0,7]
  • b为结构体变量且长度大小为24,根据规则二我们得出b不能从24开始,所以b的占位序号为[8,31]
  • c的长度为1,占位序号为[32],所以总大小为:32+1=33,然后33并不是结构体BB的最大对齐数(8)的整数倍,因此BB大小为ceil(33/8.0)*8=40.

3.IOS中对象内存大小

前面说完了结构体内存占用大小的情况,下面说说OC对象占用大小和实际申请大小该怎么计算吧。
首页,我们往Person类中新增name,nickName,age等几个属性.

#import <Foundation/Foundation.h>
@interface Person : NSObject
@property (nonatomic,copy)NSString *name;
@property (nonatomic,copy)NSString *nickName;
@property (nonatomic,assign)unsigned int age;
@property (nonatomic,assign)double score;
@end

并赋值如下:

#import <Foundation/Foundation.h>
#import "Person.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        NSLog(@"Hello, World!");
        
        Person *per = [[Person alloc] init];
        per.name = @"James";
        per.nickName = @"Potter";
        per.age = 18;
        NSLog(@"per:%@",per);
        NSLog(@"需要申请大小:%ld",class_getInstanceSize([per class]));//40
        NSLog(@"实际申请大小:%ld",malloc_size((__bridge const void *)per));//48
        
    }
    return 0;
}

因为Person有四个属性:name和nickName是指针变量各占用8个字节,age占用4个字节,score占用8个字节。其次通过结构体我们可以前面clang编译我们可以看到,结构体里面还有一个isa指针,所以Person类总大小为:8+8+8+4+8=36,对齐后应该是最大长度的整数倍,即为40.
然而为什么在实际申请内存过程中是48呢?其实苹果底层在申请内存是是按照16字节来申请的.
通过objc4中class_getInstanceSize源码分析

/** 
 * Returns the size of instances of a class.
 * 
 * @param cls A class object.
 * 
 * @return The size in bytes of instances of the class \e cls, or \c 0 if \e cls is \c Nil.
 */
OBJC_EXPORT size_t
class_getInstanceSize(Class _Nullable cls) 
    OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);


size_t class_getInstanceSize(Class cls)
{
    if (!cls) return 0;
    return cls->alignedInstanceSize();
}


// Class's ivar size rounded up to a pointer-size boundary.
uint32_t alignedInstanceSize() const {
    return word_align(unalignedInstanceSize());
}


static inline uint32_t word_align(uint32_t x) {
    //x+7 & (~7) --> 8字节对齐
    return (x + WORD_MASK) & ~WORD_MASK;
}


//其中 WORD_MASK 为
#   define WORD_MASK 7UL

实际申请内存时instanceSize源码分析

size_t instanceSize(size_t extraBytes) const {
    //编译器快速计算内存大小
    if (fastpath(cache.hasFastInstanceSize(extraBytes))) {
        return cache.fastInstanceSize(extraBytes);
    }
    
    // 计算类中所有属性的大小 + 额外的字节数0
    size_t size = alignedInstanceSize() + extraBytes;
    // CF requires all objects be at least 16 bytes.
    //如果size 小于 16,最小取16
    if (size < 16) size = 16;
    return size;
}

所以为什么苹果要按照16字节对齐呢?

  • 通常内存是由一个个字节组成的,cpu在存取数据时,并不是以字节为单位存储,而是以为单位存取,块的大小为内存存取力度。频繁存取字节未对齐的数据,会极大降低cpu的性能,所以可以通过减少存取次数来降低cpu的开销。
  • 由于在一个对象中,第一个属性isa占8字节,当然一个对象肯定还有其他属性,当无属性时,会预留8字节,即16字节对齐,如果不预留,相当于这个对象的isa和其他对象的isa紧挨着,容易造成访问混乱。
  • 16字节对齐后,可以加快CPU读取速度,同时使访问更安全,不会产生访问混乱的情况

以上就是我对对象内存大小占用情况的简单分析,欢迎各位大佬们点赞。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351