本文参考《小码哥底层原理学习视频》。
OC的代码,底层都是C/C++写的。下面我们可以看一下整个转化过程大致分为底下四个步骤:
我们暂时别看机器语言和汇编语言。我们只看到C和C++这里。OC语言既然是一门面相对象的编程语言。那么其面向对象的实现是基于C/C++的数据结构实现的呢?
首先我们先创建一个NSObject实例对象。
其实用官方的话讲上面这句代码是可以这样解释的:在内存中分配了一个16B的储存空间用来存放NSObject这个对象。再用obj这个指针指向这个对象所在的内存空间。这个对象只有一个成员变量也就是isa指针。这个成员变量占用的内存大小是8B。
为什么是是16B呢?从何得来,系统有这样的一个框架
#import<malloc/malloc.h>
这个框架提供这样一个方法:
可以打印指针所分配的内存大小。同时我们也可以点进去如下图,里面还有一步操作就是当内存不满16B的时候返回16B。
为什么又说这个对象的成员变量只有8B呢
这个时候我们用到了运行时。
导入下方这个框架:
#import <objc/runtime.h>
// 大家同样可以点进去看,苹果那边给出的注释就是获取实力对象成员所占有内存的大小。
言归正传。我们可以把这个NSObject点击去,看看内存系统是怎么实现的。
接下来我们再把这个代码转成C++代码。
怎么转换呢?大家可以百度,也可以用命令
最终我们发现类的定义就是C/C++的结构体,其中Class为结构体指针
其实大家也可以看苹果的源码。(opensource.apple.com/tarballs)
接下来我们来探讨一下如果查看一个对象的内存地址,以及如何读取对象内存和修改对象的内存。我简单的总结了一下如下图:
既然知道了NSObject对象的本质,接下来我们升级一下难度来了解一下Student对象的内存方面。请看下面一个问题:
假如我们现在有一个Student对象,它继承于NSObect对象,它有2个int类型的成员变量分别为age和no。我这边可以告诉大家在64位操作系统,一个int成员变量占用的是4B。现在问题了这个Student占用多少内存空间呢? 16B? 32B?
接下来我们再把难度升级一下:目前我们有一个person对象,person对象继承于NSObjct。person对象有一个int类型的成员变量叫做age。然后我们还有一个对象叫做Student,它继承于person。它自身有一个成员变量叫做no。问题就是一个person对象占用多少内存空间?一个student占用多少内存空间?我这里说一个答案大家相信不?16B 16B 下面我们正式进入分析问题阶段。
接下来我们看看Person对象所占用的内存:
接下来我们看看Student对象所占用的内存:
接下来我们再把难度升级一下,我现在同样有一个Person对象,它有一个int类型的成员变量叫做age。其次它还有一个int类型的属性叫做height。那么此时占用多少内存?
int类型占用的四个字节。Bool类型占用的是一个字节。char类型占用的也是一个字节。float是4个字节。double是8个字节。具体占用多少字节可以用sizeof来进行测算,比如:size_t size = sizeof(int);sizeof主要编译的时候直接查看的类型的大小。
我们继续查看源码
再继续点进去
继续点进去
实际上我们这里size就等于24然后我们再点进去,发现点不了,我们这个时候就需要去苹果下载并且查看源码。(opensource.apple.com/tarballs)
我们找到libmalloc源码下载 然后我们继续找
然后继续找
内存分配都是遵循内存对齐的原则的。我们再继续找
会发现这个定义说明了一点,内存分配的时候都是按照16的倍数进行分配的。