美文网首页
开辟内存空间

开辟内存空间

作者: redye | 来源:发表于2020-01-08 19:33 被阅读0次

    实例对象的内藏所占大小计算完成后,接下来要做的就是开辟内存空间了。开辟内存空间的源码在 libmalloc

    我们创建一个对象:

    @interface SMPerson : NSObject
    
    @property (nonatomic, copy) NSString *name;
    @property (nonatomic, copy) NSString *address;
    @property (nonatomic, assign) int age;
    @property (nonatomic, assign) long height;
    @property (nonatomic, assign) char ch;
    
    @end
    

    在 main.m 文件中

    int main(int argc, const char * argv[]) {
        @autoreleasepool {
            SMPerson *person = [[SMPerson alloc] init];
            person.name = @"CC";
            person.age = 18;
            person.address = @"上海";
            person.height = 180;
            NSLog(@"\nperson 对象所占空间大小:%lu\nperson 对象分配内存空间大小:%lu",
                  class_getInstanceSize([person class]),
                  malloc_size((void *)person));
        }
        return 0;
    }
    

    控制台打印的值为多少呢?依据上一节我们讲到内存对齐,对象创建所需要的内存空间应该是 40

    calloc.png

    对象所需要的内存的确是40字节,但是对象所开辟的空间并不一定就等于此。但是这个原因什么的,这就需要我们对calloc方法进一步探索。

    我们已经知道对象创建需要40字节的内存空间,那么就可以直接分配40字节的内存空间,即

    void *p = calloc(1, 40);
    

    跟踪方法调用的过程,最后来到:

    slot_bytes.png

    因为我们此行关注的点在于内存空间的大小,所以我们重点看方法segregated_size_to_fit

    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;
    }
    
    #define SHIFT_NANO_QUANTUM      4
    #define NANO_REGIME_QUANTA_SIZE (1 << SHIFT_NANO_QUANTUM) // 16
    

    看到这个方法是不是很眼熟 🙂

    我们将变量和宏定义都换算成数字

    k = (40 + 16 - 1) >> 4
    slot_bytes = k << 4
    

    slot_bytes 是不是就是对 size16 字节对齐,换言之,内存在开辟空间时,空间的大小都是 16 字节的倍数,每个对象的内存其实地址也都是 16字节的倍数。

    到这,是不是就解开了上面为什么 SMPerson 对象开辟的内存空间是 48 而不是 40 了。

    我们可以得出以下结论:

    • 对象创建时所需的内存空间是 8 的倍数
    • 对象创建时开辟的内存空间是 16 的倍数

    这样是不是也能解释为什么对象在创建对象是,计算出来的内存空间至少是 16 字节呢?当然这只是我的猜想,因为就算上面指定至少为 16 字节,也并不能再开辟内存空间时减少计算的时间。

    来一张 calloc 的流程图:

    calloc.jpg

    相关文章

      网友评论

          本文标题:开辟内存空间

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