自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

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

移動(dòng)開發(fā) iOS
經(jīng)過各種分析,我們可以得到的結(jié)論是 instanceSize 是以 8 字節(jié)進(jìn)行對(duì)齊的, 后面 calloc 是以 16 字節(jié)進(jìn)行對(duì)齊的,說明 calloc 進(jìn)一步對(duì)對(duì)象進(jìn)行了處理。也就解釋了我們打印出來的 40-48 了。

[[409899]]

本文轉(zhuǎn)載自微信公眾號(hào)「網(wǎng)羅開發(fā)」,作者just東東。轉(zhuǎn)載本文請(qǐng)聯(lián)系網(wǎng)羅開發(fā)公眾號(hào)。

問題的引入

初始化一個(gè) OC 類,具有如下屬性:

  1. #import <Foundation/Foundation.h> 
  2.  
  3. NS_ASSUME_NONNULL_BEGIN 
  4.  
  5. @interface LGTeacher : NSObject 
  6. @property (nonatomic, copy) NSString *name
  7. @property (nonatomic, assign) int age; 
  8. @property (nonatomic, assign) long height; 
  9. @property (nonatomic, strong) NSString *hobby; 
  10.  
  11. @end 
  12.  
  13. NS_ASSUME_NONNULL_END 

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

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

打印結(jié)果:

由以上打印結(jié)果可以看出 class_getInstanceSize 和 malloc_size 獲取到的內(nèi)存大小不一樣,那么是什么導(dǎo)致的兩者獲取同一對(duì)象的內(nèi)存大小不一樣呢?我們下一步繼續(xù)探索。

首先我們先手動(dòng)計(jì)算一下這個(gè)對(duì)象所占的內(nèi)存:

  • isa -- 8字節(jié)
  • name -- 8字節(jié)
  • age -- 4字節(jié)
  • height -- 8字節(jié)
  • hobby -- 8字節(jié)
  • 總計(jì) 36 字節(jié)。

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

  • instanceSize 繼續(xù)跟蹤

1.instanceSize

  1. size_t instanceSize(size_t extraBytes) const { 
  2.         if (fastpath(cache.hasFastInstanceSize(extraBytes))) { 
  3.             return cache.fastInstanceSize(extraBytes); 
  4.         } 
  5.  
  6.         size_t size = alignedInstanceSize() + extraBytes;// alignedInstanceSize 
  7.         // CF requires all objects be at least 16 bytes. 
  8.         if (size < 16) size = 16; 
  9.         return size
  10.  
  11. uint32_t alignedInstanceSize() const { 
  12.         return word_align(unalignedInstanceSize()); 
  13.  
  14. #   define WORD_MASK 7UL 
  15. static inline uint32_t word_align(uint32_t x) { 
  16.     return (x + WORD_MASK) & ~WORD_MASK; 

由以上源碼可以得到 instanceSize 使用 8 字節(jié)對(duì)齊原則處理 Size,并且最小為 16 字節(jié)。

2.calloc

由于 calloc 屬于 malloc 源碼里面

跟蹤 libmalloc 源碼:

calloc 源碼實(shí)現(xiàn):

  1. void * 
  2. calloc(size_t num_items, size_t size
  3.     void *retval; 
  4.     retval = malloc_zone_calloc(default_zone, num_items, size); 
  5.     if (retval == NULL) { 
  6.         errno = ENOMEM; 
  7.     } 
  8.     return retval; 
  9.  
  10. // malloc_zone_calloc 
  11. void * 
  12.     malloc_zone_calloc(malloc_zone_t *zone, size_t num_items, size_t size
  13.     MALLOC_TRACE(TRACE_calloc | DBG_FUNC_START, (uintptr_t)zone, num_items, size, 0); 
  14.  
  15.     void *ptr; 
  16.     if (malloc_check_start && (malloc_check_counter++ >= malloc_check_start)) { 
  17.         internal_check(); 
  18.     } 
  19.  
  20.     ptr = zone->calloc(zone, num_items, size); 
  21.      
  22.     if (malloc_logger) { 
  23.         malloc_logger(MALLOC_LOG_TYPE_ALLOCATE | MALLOC_LOG_TYPE_HAS_ZONE | MALLOC_LOG_TYPE_CLEARED, (uintptr_t)zone, 
  24.                 (uintptr_t)(num_items * size), 0, (uintptr_t)ptr, 0); 
  25.     } 
  26.  
  27.     MALLOC_TRACE(TRACE_calloc | DBG_FUNC_END, (uintptr_t)zone, num_items, size, (uintptr_t)ptr); 
  28.     return ptr; 

斷點(diǎn)打印 zone->calloc

  • ①:得到其真實(shí)調(diào)用為 default_zone_calloc
  • ②:搜索 default_zone_calloc 繼續(xù)跟進(jìn),打印 default_zone_calloc 內(nèi)部的 zone->calloc 得到 nano_calloc
  • ③:分析 nano_calloc 源碼可以知道在 _nano_malloc_check_clear 內(nèi)進(jìn)行了相關(guān)操作
  1. static void * 
  2. default_zone_calloc(malloc_zone_t *zone, size_t num_items, size_t size
  3.     zone = runtime_default_zone(); 
  4.      
  5.     return zone->calloc(zone, num_items, size); 
  6.  
  7. static void * 
  8. nano_calloc(nanozone_t *nanozone, size_t num_items, size_t size
  9.     size_t total_bytes; 
  10.  
  11.     if (calloc_get_size(num_items, size, 0, &total_bytes)) { 
  12.         return NULL
  13.     } 
  14.  
  15.     if (total_bytes <= NANO_MAX_SIZE) { 
  16.         void *p = _nano_malloc_check_clear(nanozone, total_bytes, 1); 
  17.         if (p) { 
  18.             return p; 
  19.         } else { 
  20.             /* FALLTHROUGH to helper zone */ 
  21.         } 
  22.     } 
  23.     malloc_zone_t *zone = (malloc_zone_t *)(nanozone->helper_zone); 
  24.     return zone->calloc(zone, 1, total_bytes); 

跳轉(zhuǎn)到 _nano_malloc_check_clear 內(nèi)部發(fā)現(xiàn)代碼很多,一臉懵逼,但是仔細(xì)一看很多都是做一些容錯(cuò)判斷,除去這些代碼后,返現(xiàn)與size 有關(guān)的只有一行代碼:

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

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

  1. #define SHIFT_NANO_QUANTUM      4 
  2. #define NANO_REGIME_QUANTA_SIZE (1 << SHIFT_NANO_QUANTUM)   // 16 
  3.  
  4. static MALLOC_INLINE size_t 
  5. segregated_size_to_fit(nanozone_t *nanozone, size_t size, size_t *pKey) 
  6.     size_t k, slot_bytes; 
  7.  
  8.     if (0 == size) { 
  9.         size = NANO_REGIME_QUANTA_SIZE; // Historical behavior 
  10.     } 
  11.     k = (size + NANO_REGIME_QUANTA_SIZE - 1) >> SHIFT_NANO_QUANTUM; // round up and shift for number of quanta 
  12.     slot_bytes = k << SHIFT_NANO_QUANTUM;                           // multiply by power of two quanta size 
  13.     *pKey = k - 1;                                                  // Zero-based! 
  14.  
  15.     return slot_bytes; 

總結(jié)

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

 

由以上可以知道對(duì)象申請(qǐng)的內(nèi)存大小和系統(tǒng)開辟的大小存在不一致的情況,8 字節(jié)對(duì)齊應(yīng)用于對(duì)象的屬性,16 字節(jié)對(duì)齊應(yīng)用于對(duì)象,由于對(duì)象的內(nèi)存是連續(xù)的,這樣可以規(guī)避一些不必要的風(fēng)險(xiǎn),以空間換時(shí)間來得到更高的安全性。

 

責(zé)任編輯:武曉燕 來源: 網(wǎng)羅開發(fā)
相關(guān)推薦

2021-08-06 11:50:49

Linux 字節(jié)對(duì)齊Linux 系統(tǒng)

2021-08-16 06:56:21

Slice數(shù)組類型內(nèi)存

2021-12-16 06:52:33

C語言內(nèi)存分配

2022-11-30 08:19:15

內(nèi)存分配Go逃逸分析

2021-01-07 07:53:10

JavaScript內(nèi)存管理

2018-03-27 10:06:26

對(duì)象存儲(chǔ)演進(jìn)

2013-04-17 10:46:54

面向?qū)ο?/a>

2022-11-28 07:21:53

操作系統(tǒng)內(nèi)存管理

2023-06-14 08:15:34

算法合并操作Winner

2024-03-05 10:09:16

restfulHTTPAPI

2010-08-10 10:00:57

Flex內(nèi)存

2012-06-07 10:11:01

面向?qū)ο?/a>設(shè)計(jì)原則Java

2024-12-09 08:18:33

2021-02-03 15:12:08

java內(nèi)存溢出

2021-05-19 08:04:11

ASP.Net服務(wù)性原則

2009-10-09 09:42:07

虛擬機(jī)內(nèi)存

2019-07-11 15:43:44

KVMKSM內(nèi)存

2020-05-09 13:49:00

內(nèi)存空間垃圾

2020-09-25 07:55:51

內(nèi)存Java對(duì)象

2022-12-12 08:42:06

Java對(duì)象棧內(nèi)存
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)