当前位置:首页 >时尚 >聊聊iOS OC 对象的内存对齐原则 8 字节对齐应用于对象的属性

聊聊iOS OC 对象的内存对齐原则 8 字节对齐应用于对象的属性

2024-06-30 18:12:22 [百科] 来源:避面尹邢网

聊聊iOS OC 对象的聊聊内存对齐原则

作者: just东东 移动开发 iOS 经过各种分析,我们可以得到的对对齐结论是 instanceSize 是以 8 字节进行对齐的, 后面 calloc 是内存以 16 字节进行对齐的,说明 calloc 进一步对对象进行了处理。原则也就解释了我们打印出来的聊聊 40-48 了。

[[409899]]

本文转载自微信公众号「网罗开发」,对对齐作者just东东。内存转载本文请联系网罗开发公众号。原则

问题的聊聊引入

初始化一个 OC 类,具有如下属性:

聊聊iOS OC 对象的内存对齐原则 8 字节对齐应用于对象的属性

  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 

初始化对象,并获取对象的内存内存 size:

聊聊iOS OC 对象的内存对齐原则 8 字节对齐应用于对象的属性

  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))); 

打印结果:

聊聊iOS OC 对象的内存对齐原则 8 字节对齐应用于对象的属性

由以上打印结果可以看出 class_getInstanceSize 和 malloc_size 获取到的内存大小不一样,那么是原则什么导致的两者获取同一对象的内存大小不一样呢?我们下一步继续探索。

首先我们先手动计算一下这个对象所占的聊聊内存:

  • isa -- 8字节
  • name -- 8字节
  • age -- 4字节
  • height -- 8字节
  • hobby -- 8字节
  • 总计 36 字节。

我们跟踪 objc 源码可以发现改变 size 的对对齐地方有两个地方:

  • instanceSize 继续跟踪

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 字节对齐原则处理 Size,并且最小为 16 字节。内存

2.calloc

由于 calloc 属于 malloc 源码里面

跟踪 libmalloc 源码:

calloc 源码实现:

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

断点打印 zone->calloc

  • ①:得到其真实调用为 default_zone_calloc
  • ②:搜索 default_zone_calloc 继续跟进,打印 default_zone_calloc 内部的 zone->calloc 得到 nano_calloc
  • ③:分析 nano_calloc 源码可以知道在 _nano_malloc_check_clear 内进行了相关操作
  1. static void * 
  2. default_zone_calloc(malloc_zone_t *zone, size_t num_items, size_t size) 
  3. {  
  4.     zone = runtime_default_zone(); 
  5.      
  6.     return zone->calloc(zone, num_items, size); 
  7.  
  8. static void * 
  9. nano_calloc(nanozone_t *nanozone, size_t num_items, size_t size) 
  10. {  
  11.     size_t total_bytes; 
  12.  
  13.     if (calloc_get_size(num_items, size, 0, &total_bytes)) {  
  14.         return NULL; 
  15.     } 
  16.  
  17.     if (total_bytes <= NANO_MAX_SIZE) {  
  18.         void *p = _nano_malloc_check_clear(nanozone, total_bytes, 1); 
  19.         if (p) {  
  20.             return p; 
  21.         } else {  
  22.             /* FALLTHROUGH to helper zone */ 
  23.         } 
  24.     } 
  25.     malloc_zone_t *zone = (malloc_zone_t *)(nanozone->helper_zone); 
  26.     return zone->calloc(zone, 1, total_bytes); 

跳转到 _nano_malloc_check_clear 内部发现代码很多,一脸懵逼,但是仔细一看很多都是做一些容错判断,除去这些代码后,返现与size 有关的只有一行代码:

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

跳转进 segregated_size_to_fit 可以看到又是内存对齐的代码,这里的内存对齐是以16字节原则进行对齐的。

  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. {  
  7.     size_t k, slot_bytes; 
  8.  
  9.     if (0 == size) {  
  10.         size = NANO_REGIME_QUANTA_SIZE; // Historical behavior 
  11.     } 
  12.     k = (size + NANO_REGIME_QUANTA_SIZE - 1) >> SHIFT_NANO_QUANTUM; // round up and shift for number of quanta 
  13.     slot_bytes = k << SHIFT_NANO_QUANTUM;                           // multiply by power of two quanta size 
  14.     *pKey = k - 1;                                                  // Zero-based! 
  15.  
  16.     return slot_bytes; 

总结

经过上述的各种分析,我们可以得到的结论是 instanceSize 是以 8 字节进行对齐的, 后面 calloc 是以 16 字节进行对齐的,说明 calloc 进一步对对象进行了处理。也就解释了我们打印出来的 40-48 了。

 

由以上可以知道对象申请的内存大小和系统开辟的大小存在不一致的情况,8 字节对齐应用于对象的属性,16 字节对齐应用于对象,由于对象的内存是连续的,这样可以规避一些不必要的风险,以空间换时间来得到更高的安全性。

 

责任编辑:武晓燕 来源: 网罗开发 iOS内存对齐

(责任编辑:热点)

    推荐文章
    热点阅读