iOS OC 對象的內(nèi)存對齊原則

iOS OC 對象的內(nèi)存對齊原則

1.問題的引入

初始化一個OC類得湘,具有如下屬性:

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface LGTeacher : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@property (nonatomic, assign) long height;
@property (nonatomic, strong) NSString *hobby;

@end

NS_ASSUME_NONNULL_END

初始化對象惋戏,并獲取對象的內(nèi)存size:

        LGTeacher  *p = [[LGTeacher alloc] init];
        p.name = @"LG_Cooci";
        p.age  = 18;
        p.height = 185;
        p.hobby  = @"女";
        
        NSLog(@"%lu - %lu",class_getInstanceSize([p class]),malloc_size((__bridge const void *)(p)));

打印結果:

image

由以上打印結果可以看出 class_getInstanceSizemalloc_size獲取到的內(nèi)存大小不一樣火惊,那么是什么導致的兩者獲取同一對象的內(nèi)存大小不一樣呢唯欣?我們下一步繼續(xù)探索居触。

首先我們先手動計算一下這個對象所占的內(nèi)存:
isa -- 8字節(jié)拜秧,name -- 8字節(jié)痹屹, age -- 4字節(jié), height -- 8字節(jié)枉氮, hobby -- 8字節(jié)志衍;總計36字節(jié)。

我們跟蹤objc源碼可以發(fā)現(xiàn)改變size的地方有兩個地方:

image
    1. instanceSize
      instanceSize 繼續(xù)跟蹤聊替,
size_t instanceSize(size_t extraBytes) const {
        if (fastpath(cache.hasFastInstanceSize(extraBytes))) {
            return cache.fastInstanceSize(extraBytes);
        }

        size_t size = alignedInstanceSize() + extraBytes;// alignedInstanceSize
        // CF requires all objects be at least 16 bytes.
        if (size < 16) size = 16;
        return size;
}

uint32_t alignedInstanceSize() const {
        return word_align(unalignedInstanceSize());
}

#   define WORD_MASK 7UL
static inline uint32_t word_align(uint32_t x) {
    return (x + WORD_MASK) & ~WORD_MASK;
}

由以上源碼可以得到instanceSize 使用8字節(jié)對齊原則處理Size楼肪,并且最小為16字節(jié)。
其中的原理可以參考本人其他篇文章:內(nèi)存對齊小記佃牛,內(nèi)存對齊算法淹辞。

    1. calloc

由于calloc屬于malloc源碼里面

跟蹤libmalloc源碼:

calloc源碼實現(xiàn):

void *
calloc(size_t num_items, size_t size)
{
    void *retval;
    retval = malloc_zone_calloc(default_zone, num_items, size);
    if (retval == NULL) {
        errno = ENOMEM;
    }
    return retval;
}

// malloc_zone_calloc
void *
    malloc_zone_calloc(malloc_zone_t *zone, size_t num_items, size_t size)
{
    MALLOC_TRACE(TRACE_calloc | DBG_FUNC_START, (uintptr_t)zone, num_items, size, 0);

    void *ptr;
    if (malloc_check_start && (malloc_check_counter++ >= malloc_check_start)) {
        internal_check();
    }

    ptr = zone->calloc(zone, num_items, size);
    
    if (malloc_logger) {
        malloc_logger(MALLOC_LOG_TYPE_ALLOCATE | MALLOC_LOG_TYPE_HAS_ZONE | MALLOC_LOG_TYPE_CLEARED, (uintptr_t)zone,
                (uintptr_t)(num_items * size), 0, (uintptr_t)ptr, 0);
    }

    MALLOC_TRACE(TRACE_calloc | DBG_FUNC_END, (uintptr_t)zone, num_items, size, (uintptr_t)ptr);
    return ptr;
}

斷點打印 zone->calloc

  • ①:得到其真實調(diào)用為default_zone_calloc
  • ②:搜索default_zone_calloc繼續(xù)跟進,打印default_zone_calloc內(nèi)部的zone->calloc得到 nano_calloc
  • ③:分析nano_calloc源碼可以知道在 _nano_malloc_check_clear內(nèi)進行了相關操作
static void *
default_zone_calloc(malloc_zone_t *zone, size_t num_items, size_t size)
{
    zone = runtime_default_zone();
    
    return zone->calloc(zone, num_items, size);
}

static void *
nano_calloc(nanozone_t *nanozone, size_t num_items, size_t size)
{
    size_t total_bytes;

    if (calloc_get_size(num_items, size, 0, &total_bytes)) {
        return NULL;
    }

    if (total_bytes <= NANO_MAX_SIZE) {
        void *p = _nano_malloc_check_clear(nanozone, total_bytes, 1);
        if (p) {
            return p;
        } else {
            /* FALLTHROUGH to helper zone */
        }
    }
    malloc_zone_t *zone = (malloc_zone_t *)(nanozone->helper_zone);
    return zone->calloc(zone, 1, total_bytes);
}

跳轉到_nano_malloc_check_clear內(nèi)部發(fā)現(xiàn)代碼很多俘侠,一臉懵逼象缀,但是仔細一看很多都是做一些容錯判斷,除去這些代碼后爷速,返現(xiàn)與size有關的只有一行代碼:

size_t slot_bytes = segregated_size_to_fit(nanozone, size, &slot_key);

跳轉進 segregated_size_to_fit 可以看到又是內(nèi)存對齊的代碼央星,這里的內(nèi)存對齊是以16字節(jié)原則進行對齊的。

內(nèi)存對齊的原理可以參考本人其他篇文章:內(nèi)存對齊小記惫东,內(nèi)存對齊算法莉给。

#define SHIFT_NANO_QUANTUM      4
#define NANO_REGIME_QUANTA_SIZE (1 << SHIFT_NANO_QUANTUM)   // 16

static MALLOC_INLINE size_t
segregated_size_to_fit(nanozone_t *nanozone, size_t size, size_t *pKey)
{
    size_t k, slot_bytes;

    if (0 == size) {
        size = NANO_REGIME_QUANTA_SIZE; // Historical behavior
    }
    k = (size + NANO_REGIME_QUANTA_SIZE - 1) >> SHIFT_NANO_QUANTUM; // round up and shift for number of quanta
    slot_bytes = k << SHIFT_NANO_QUANTUM;                           // multiply by power of two quanta size
    *pKey = k - 1;                                                  // Zero-based!

    return slot_bytes;
}

總結

經(jīng)過上述的各種分析毙石,我們可以得到的結論是instanceSize是以8字節(jié)進行對齊的, 后面calloc是以16字節(jié)進行對齊的颓遏,說明calloc進一步對對象進行了處理徐矩。也就解釋了我們打印出來的40-48了。

由以上可以知道對象申請的內(nèi)存大小和系統(tǒng)開辟的大小存在不一致的情況叁幢,8字節(jié)對齊應用于對象的屬性滤灯,16字節(jié)對齊應用于對象,由于對象的內(nèi)存是連續(xù)的曼玩,這樣可以規(guī)避一些不必要的風險鳞骤,以空間換時間來得到更高的安全性。

?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末黍判,一起剝皮案震驚了整個濱河市豫尽,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌顷帖,老刑警劉巖美旧,帶你破解...
    沈念sama閱讀 206,214評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異贬墩,居然都是意外死亡陈症,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評論 2 382
  • 文/潘曉璐 我一進店門震糖,熙熙樓的掌柜王于貴愁眉苦臉地迎上來录肯,“玉大人,你說我怎么就攤上這事吊说÷塾剑” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評論 0 341
  • 文/不壞的土叔 我叫張陵颁井,是天一觀的道長厅贪。 經(jīng)常有香客問我,道長雅宾,這世上最難降的妖魔是什么养涮? 我笑而不...
    開封第一講書人閱讀 55,221評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮眉抬,結果婚禮上贯吓,老公的妹妹穿的比我還像新娘。我一直安慰自己蜀变,他們只是感情好悄谐,可當我...
    茶點故事閱讀 64,224評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著库北,像睡著了一般爬舰。 火紅的嫁衣襯著肌膚如雪们陆。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評論 1 284
  • 那天情屹,我揣著相機與錄音坪仇,去河邊找鬼。 笑死垃你,一個胖子當著我的面吹牛烟很,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蜡镶,決...
    沈念sama閱讀 38,313評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼恤筛!你這毒婦竟也來了官还?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 36,956評論 0 259
  • 序言:老撾萬榮一對情侶失蹤毒坛,失蹤者是張志新(化名)和其女友劉穎望伦,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體煎殷,經(jīng)...
    沈念sama閱讀 43,441評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡屯伞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,925評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了豪直。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片劣摇。...
    茶點故事閱讀 38,018評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖弓乙,靈堂內(nèi)的尸體忽然破棺而出末融,到底是詐尸還是另有隱情,我是刑警寧澤暇韧,帶...
    沈念sama閱讀 33,685評論 4 322
  • 正文 年R本政府宣布勾习,位于F島的核電站,受9級特大地震影響懈玻,放射性物質發(fā)生泄漏巧婶。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,234評論 3 307
  • 文/蒙蒙 一涂乌、第九天 我趴在偏房一處隱蔽的房頂上張望艺栈。 院中可真熱鬧,春花似錦湾盒、人聲如沸眼滤。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽诅需。三九已至漾唉,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間堰塌,已是汗流浹背赵刑。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留场刑,地道東北人般此。 一個月前我還...
    沈念sama閱讀 45,467評論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像牵现,于是被迫代替她去往敵國和親铐懊。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,762評論 2 345

推薦閱讀更多精彩內(nèi)容