美文网首页
OC instance 对象本质三

OC instance 对象本质三

作者: 曹来东 | 来源:发表于2018-08-12 14:31 被阅读28次
@interface LDPerson : NSObject
{
    int _no;
    int _age;
}
@end

LDPerson底层实现:

struct NSObject_IMPL{
    Class isa;
}
struct LDPerson_IMPL
{
    struct NSObject_IMPL NSObject_IVARS;//8字节
    int _no;//4字节
    int _age;//4字节
}

所以上图中LDPersoninstance对象占用16字节的内存.

思考下图中LDPersoninstance对象占用的内存大小

@interface LDPerson : NSObject
{
    int _age;
    int _height;
    int _no;
}

底层实现:

struct NSObject_IMPL{
    Class isa;
}
struct LDPerson_IMPL
{
    struct NSObject_IMPL NSObject_IVARS;//8字节
    int _no;//4字节
    int _height;//4字节
    int _age;//4字节
}

推断:实际占用20字节,因为内存对其原则,必须为最大成员变量 所占内存的整数倍.所以该instance对象应该占用24个字节.

通过代码验证:

image.png
通过代码我们可以发现:之前的推论是错误的.LDPersoninstance对象实际占用24字节,分配时占用32字节.

通过内存图分析:

image.png
如图红框部分一般为连续内存地址,即为LDPerson的instance对象.可以看出占用32个字节.

通过OC代码转成C/C++代码再次查看

1. cd 目标路径
2.xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m -o main-arm64.cpp

查看后得知LDPerson底层实现结构就如我们上面所写,所以LDPerson确实占用24字节,分配32字节.

查看objc4开源代码

  1. 检索allocWithZone 查看.mm文件
id
_objc_rootAllocWithZone(Class cls, malloc_zone_t *zone)
{
    id obj;

#if __OBJC2__
    // allocWithZone under __OBJC2__ ignores the zone parameter
    (void)zone;
    obj = class_createInstance(cls, 0);
#else
    if (!zone) {
        obj = class_createInstance(cls, 0);
    }
    else {
        obj = class_createInstanceFromZone(cls, 0, zone);
    }
#endif

    if (slowpath(!obj)) obj = callBadAllocHandler(cls);
    return obj;
}

主要代码为:


image.png
id 
class_createInstance(Class cls, size_t extraBytes)
{
    return _class_createInstanceFromZone(cls, extraBytes, nil);
}

可以看到extraBytes传入的值为0.再看_class_createInstanceFromZone的实现:

id
_class_createInstanceFromZone(Class cls, size_t extraBytes, void *zone, 
                              bool cxxConstruct = true, 
                              size_t *outAllocatedSize = nil)
{
    if (!cls) return nil;

    assert(cls->isRealized());

    // Read class's info bits all at once for performance
    bool hasCxxCtor = cls->hasCxxCtor();
    bool hasCxxDtor = cls->hasCxxDtor();
    bool fast = cls->canAllocNonpointer();

    size_t size = cls->instanceSize(extraBytes);
    if (outAllocatedSize) *outAllocatedSize = size;

    id obj;
    if (!zone  &&  fast) {
        obj = (id)calloc(1, size);
        if (!obj) return nil;
        obj->initInstanceIsa(cls, hasCxxDtor);
    } 
    else {
        if (zone) {
            obj = (id)malloc_zone_calloc ((malloc_zone_t *)zone, 1, size);
        } else {
            obj = (id)calloc(1, size);
        }
        if (!obj) return nil;

        // Use raw pointer isa on the assumption that they might be 
        // doing something weird with the zone or RR.
        obj->initIsa(cls);
    }

    if (cxxConstruct && hasCxxCtor) {
        obj = _objc_constructOrFree(obj, cls);
    }

    return obj;
}
image.png
所以我们可以看到,实际分配内存大小就在instanceSizecalloc两个函数
size_t instanceSize(size_t extraBytes) {
        size_t size = alignedInstanceSize() + extraBytes;
        // CF requires all objects be at least 16 bytes.
        if (size < 16) size = 16;
        return size;
    }
 // May be unaligned depending on class's ivars.
    uint32_t unalignedInstanceSize() {
        assert(isRealized());
        return data()->ro->instanceSize;
    }

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

对比size_t instanceSize(size_t extraBytes)class_getInstanceSize(Class cls)的实现

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

因为extraBytes为0所以两个函数的返回值类似,只是前者多了一个与16字节的判断.
然后我们再看obj = (id)calloc(1, size);这个函数,如下可知C语言的标准库,看不到实现的代码.

image.png

通过其他开源代码查看calloc函数的实现

  1. 打开开源代码
  2. 检索libmalloc关键字

3.libmalloc-140.40.1.tar.gz
4.按如下所示查找到calloc实现如下:

image.png

iOS堆内存 分配原则:都是16的倍数

可通过下面的方式查看:

image.png
class_getInstanceSize():
创建一个实例对象,至少需要多少内存对齐后.
malloc_size()
创建一个实例对象,实际上分配多少内存.

相关文章

网友评论

      本文标题:OC instance 对象本质三

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