OC对象的本质

OC对象的本质

  • 我们平常编写的 Objective-C 代码,底层实现其实都是 C/C++ 代码
  • 具体的实现过程,就是 Objective-C ——>C/C++———>汇编语言———>机器语言
底层实现
  • 注意所以Objective-C的面向对象都是基于C/C++的数据结构实现的

Objective-C转换成C/C++代码

创建一个命令行项目

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[]) {
   @autoreleasepool {
       NSObject *obj = [[NSObject alloc] init];
       NSLog(@"Hello, World!");
   }
   return 0;
}

在项目的目录下运行clang -rewrite-objc main.m -o main.cpp显示如下,生成.cpp文件

生成.cpp

命令:clang -rewrite-objc main.m -o main.cpp

  • clang Xcode 内置的 LLVM 编译器

  • -rewrite-objc 从写objc代码

  • main.m 源文件

  • -o 输出

  • main.cpp cpp="c plus plus" 既是C++

    但是上面的命令会输出 多平台代码,内存大(9万多行代码),不便于读取,Windows ,MAC,因此一般需要我们指定平台,用于iPhone手机例如 xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m -o main-arm64.cpp

    命令:xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc OC源文件 -o 输出文件

    与上条命令相比较多了几点:

    • xcrun Xcode run
    • -sdk
    • iphoneos 指定平台
    • -arch 架构 模拟器(i386)、32bit(armv7)、5s后都是64bit(arm64)

如果需要链接其他框架,使用-framework参数,比如 -framework UIKit

指定输出

生成的main-arm64.cpp(3万多行代码),虽然比上面那个9万多行少许多,但也看的一脸懵逼啊,全局搜索int main 函数

int main(int argc, const char * argv[]) {
 /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 

    NSObject *obj=((NSObject *(*)(id, SEL))(void *)objc_msgSend)((id)((NSObject *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("NSObject"), sel_registerName("alloc")), sel_registerName("init"));

    NSLog((NSString *)&__NSConstantStringImpl__var_folders_c0_7nm4_r7s4xd0mbs67ljb_b8m0000gn_T_main_1b47c1_mi_0);
 }
 return 0;
}

然后搜索 NSObject_IMPL 显示下面的结构体,也就是说Objective-C编译后NSObject对象会编译成下面这个结构体

struct NSObject_IMPL {
    Class isa; // 64位中占 8个字节
};
既是, NSObject *obj = [[NSObject alloc] init];
一个NSObject 对象在内存中就是上面那个结构体形式
也就是上面说的,C/C++的结构体支撑了Objective-C

在进入Class中可以看到
typedef struct objc_class *Class;
释义:Class是一个指向结构体的指针 指针在32位中占4个字节,在64位中占8个字节

然后,返回main.m文件,进入NSObject中可以看到和上面的NSObject_IMPL的结构

@interface NSObject <NSObject> {
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wobjc-interface-ivars"
    Class isa  OBJC_ISA_AVAILABILITY;
#pragma clang diagnostic pop
}

这就是我们编写的OC语言转换成C++后的代码

编译后的结构体

思考 一个OC对象在内存中是如何布局的?

  • NSObject的底层实现


    实现原理

面试题

一个NSobject对象占用多少内存

  • 导入 #import <objc/runtime.h>通过class_getInstanceSize可以查看内存
  • 但是NSObject对象内部只使用了8个字节的空间(bit64环境下,可以通过class_getInstanceSize函数获得)
  • 导入 #import <malloc/malloc.h>通过malloc_size可以查看内存
  • 系统分配了16个字节给NSobject对象(通过malloc_size函数获得)
 // 获得NSObject实例对象的成员变量所占用的大小 >> 8
NSLog(@"%zd", class_getInstanceSize([NSObject class])); 

// 获得obj指针所指向内存的大小 >> 16
NSLog(@"%zd", malloc_size((__bridge const void *)obj)); 

分析 :从下图可以看出,内存分配了16个字节,实际使用的使用8个字节存放isa


内存分配
通过查看alloc 底层也能发现,size小于16 会自动取值为16
// Class's ivar size rounded up to a pointer-size boundary.
    uint32_t alignedInstanceSize() {
        return word_align(unalignedInstanceSize());
    }

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

  • 方式二
    代码中打断点,打印obj 的内存,然后使用Xcode查看内存
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSObject *obj = [[NSObject alloc] init];
    return 0;
}

使用Xcode 的Debug来调试
Debug——>Debug Workflow ——>View Memory


Debug调试

通过打印可以看出前8个字节有值,后8个直接无值


内存分布
  • 方式三
    使用LLDB指令
 // print / p 打印指针
(lldb) print obj
(NSObject *) $0 = 0x0000000100753df0
(lldb) p obj
(NSObject *) $1 = 0x0000000100753df0
// po 打印对象
(lldb) po obj
<NSObject: 0x100753df0>
// 读取内存 memory read = x 
(lldb) memory read 0x100753df0
0x100753df0: 41 d1 a4 99 ff ff 1d 00 00 00 00 00 00 00 00 00  A...............
0x100753e00: d0 3e 75 00 01 00 00 00 10 41 75 00 01 00 00 00  .>u......Au.....

x/3xw 0x100753df0

x memory read = x
3 数量
x格式
w字节数
0x100753df0 内存地址

格式
x是16进制 f是浮点 d是10进制
字节大小
bbyte 1字节 hhalf word 2字节 wword 4字节 ggiant word 8字节
修改内存中的值
memory write 内存地址 数值
memory write 0x100753df0 10

利用一个简单的对象再次探索OC对象的本质

创建一个对象继承自NSObject

@interface Student : NSObject
{
    int _no;
    int _age;
}
@end

使用上面的命令生成.cpp文件,查看对应的关键源码

struct NSObject_IMPL {
    Class isa;
};

struct Student_IMPL {
    struct NSObject_IMPL NSObject_IVARS;  // 占8个字节 先写父类的实现,然后写自己的属性
    int _age;
    int _no;
};
// 上面两个代码等价于
struct Student_IMPL {
    Class isa; // 8个字节
    int _age; // 4个字节
    int _no; // 4个字节
};

猜想 上面的Student对象占用多少内存

  • 根据内存对齐,结构体的大小必须是最大成员大小的倍数

上面代码转换流程如下

底层实现

执行下面代码,得知,分配了16个字节空间,使用了16个字节空间

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        Student *stu = [[Student alloc] init];
        // 通过指针,直接访问成员变量
        stu->_no = 4;
        stu->_age = 5;
        
        NSLog(@"%zd", class_getInstanceSize([Student class]));
        NSLog(@"%zd", malloc_size((__bridge const void *)stu));
        
        //强制stu转化为结构体
        struct Student_IMPL *stuImpl = (__bridge struct Student_IMPL *)stu;
        NSLog(@"no is %d, age is %d", stuImpl->_no, stuImpl->_age);
    }
    return 0;
}

思考 一个Person对象,一个Student对象占用多少内存空间

// Person
@interface Person : NSObject
{
    int _age;
} 
// 16 = isa+_age = 8+4  但是根据内存对齐法则:结构体的大小必须是最大成员大小的倍数
@end
// Student
@interface Student : Person
{
    int _no;
} 
// 16  = isa+_age +_no = 8+4+4 刚好占用16个字节
@end
转换过程

内存分布图


内存分布图

再次思考 一个Person对象占用了多少内存空间

@interface Person : NSObject
{
    int _age;
    int _height;
    int _no;
}
// 根据底层实现原理 如下:
struct Person_IMPL {
    struct NSObject_IMPL NSObject_IVARS; // 8
    int _age; // 4
    int _height; // 4
    int _no; // 4
};  
// 计算结构体大小,内存对齐,应该为结构体中最大成员大小的倍数 为 24

// 打印
Person *p = [[Person alloc] init];

NSLog(@"%zd", sizeof(struct Person_IMPL)); // 24
// 因为系统alloc分配内存为16的倍数,为了内存的速度,所以为 32
NSLog(@"%zd %zd",
              class_getInstanceSize([Person class]), // 24
              malloc_size((__bridge const void *)(p))); // 32
@end

根据苹果开源库,查看得知
开源库 https://opensource.apple.com/tarballs/libmalloc/

libmalloc

2个容易混淆的函数

  • 创建一个实例对象,至少需要多少内存?
    • #import <objc/runtime.h>
      class_getInstanceSize([NSObject class])
  • 创建一个实例对象,实际上分配了多少内存?
    • #import <malloc/malloc.h>
      malloc_size((__bridge const void *) obj)

扩展

可以使用gnu来窥探,内存分配

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,219评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,363评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,933评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,020评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,400评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,640评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,896评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,597评论 0 199
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,327评论 1 244
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,581评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,072评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,399评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,054评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,083评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,849评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,672评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,585评论 2 270

推荐阅读更多精彩内容