iOS底层原理探究- NSObject 所占内存

iOS底层原理探究- NSObject 所占内存

面向对象的Objective-C

我们平时写的 OC 代码底层实现为 C/C++ 代码,因为 RuntimeOC 具备了面向对象的特点,而后底层的 C/C++ 会转换成底层的 汇编 代码,最终被被解析成计算机能识别的 机器语言 。而 OC 中的类,正是正是基于 C/C++ 的结构体实现的。我们可以通过 clang 命令将我们平时所写的 OC 代码转换为 C/C++ 代码。这是转换代码:

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc 源文件名 -o 目标文件名
复制代码

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

如: 我们进入源文件所在的目录,执行 xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m -o main.cpp 会在当前目录生成一个main.cpp的文件,这就是一个最简单的OC文件的 C++ 实现。

通过转换之后我们很容易找到 NSObject 类的真正实现:

struct NSObject_IMPL {
	Class isa;
};
复制代码

只有一个 名为 isaClass 实例。继续探寻 Class 的声明:

/// An opaque type that represents an Objective-C class.
typedef struct objc_class *Class;
复制代码

发现 Class 实际上是一个指向 objc_class 的结构体指针。

也就是说 NSObject 最终声明为一个 指向结构体 objc_class 的名为 isa 的结构体指针。既然是指针,在32位系统中占 4个 字节,在64位系统中占 8 个字节。如何查看自己的Mac是多少位呢?可以在终端 输入 uname -a 若末尾显示 x86_64 则代表是64位系统,如果末尾显示 i686 则代表是32 系统。目前我们所使用的大多都是 64 位。

使用 Runtime 中 class_getInstanceSize 输出一个类的实例对象的成员变量的大小
        NSObject *obj = [[NSObject alloc] init];
        NSLog(@"%zd",class_getInstanceSize([obj class]));
复制代码

输出:

由此可见,一个 NSObject 只有一个成员变量 即 isa,它所占 8 个字节的大小。 我们可通过 objc4源码得出此结论。

Class's ivar size rounded up to a pointer-size boundary. 一个类的所有成员变量所占用的空间。

那在 OC 中一个 NSObject 对象占用8个字节吗?答案是否定的,我们继续分析。

使用 malloc_size(const void *ptr) 输出一个类实际分配的内存大小
        NSObject *obj = [[NSObject alloc] init];
        NSLog(@"%zd",malloc_size((__bridge const void *)obj));
复制代码

输出:

2018-05-29 07:40:24.270554+0800 XWTest0[25108:1196172] 16
复制代码
此外我们也可以用这种方式来证明一个NSObject对象占 16 个字节

我们知道在 OC 的对象实例中, 真正给对象分配内存的方式是

+ (instancetype)allocWithZone:(struct _NSZone *)zone OBJC_SWIFT_UNAVAILABLE("use object initializers instead");
复制代码

我们可以探究 allocWithZone 方法的底层实现, 同样查看Apple开源的OC源码 objc4源码

所有的OC对象至少为16字节。进一步证明 NSObject类实际所占用的内存空间为 16 个字节

你可能感兴趣的:(iOS底层原理探究- NSObject 所占内存)