手把手带你去分析NSObject本质是什么

简介: 这篇行文主要讲 NSObject 对象三连问:NSObject本质是什么,占用多少内存,以及怎么去分析。从下图我们可以看到,我们平时编写的 Objective-C 代码,最终转换到机器语言的过程:https://img-blog.csdnimg.cn/2a6a930a8d3546f896d7527a95c3a576.png换句话说,Objective-C的面向对象都是基于C\C++的数据结构实现的,或者说其底层实现其实都是C\C++代码。

一、代码转换


1、 添加 NSObject 对象

如在 main.m 添加一个 NSObject 对象(或者直接复制以下代码到 main.m 文件中):


#import <Foundation/Foundation.h>
#import <objc/runtime.h>
#import <malloc/malloc.h>
int main(int argc, char * argv[]) {
    @autoreleasepool {
        NSObject *obj = [[NSObject alloc] init];
    }
    return 0;
}


2、命令行实现

已知:

1、不同平台支持的代码是不一样的( Windows、mac、iOS)

2、iOS架构:模拟器(i386)、32bit(armv7)、64bit(arm64)


命令行格式如下:


// 不指定平台和架构
clang -rewrite-objc main.m -o main.cpp
// 指定真机(ios平台64bit架构)
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m -o main-arm64.cpp
// 指定模拟器
xcrun -sdk iphonesimulator clang -arch i386 -rewrite-objc  main.m -o main-i386-simulator.cpp
// 命令行格式
// xcrun:xcode run
// -sdk 平台:iphoneos
// -arch 架构:arm64
// -rewrite-objc:重写OC
// main.m -o main-arm64-1.cpp:OC源文件  -o  输出的CPP文件
// 如果需要链接其他框架,使用-framework参数。比如-framework UIKit
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m -o main-arm64.cpp


执行生成目录:


chenzimin@rattanchen InterView % tree
.
├── Info.plist
├── main-arm64-iphoneos.cpp
├── main-i386-simulator.cpp
├── main.cpp
└── main.m
0 directories, 5 files


3、结果

main 方法中转化的代码依次如下:

1、不指定机型架构:

dd3438357fb7458ba858730b52770c38.png

2、指定机型架构:

e0b471c205eb4253aacf31546f2216bf.png

以上直观的感知转cpp的代码是一样的,但是不配置机型机构的部分,因为需要涉及兼容问题,代码行数相差了将近三倍。所以按机型配置可以节省很多存储空间。


二、本质分析


看到OC实现和 C\C++ 底层实现,如下:

03fbf95dd7fe4a27ab6ce7b00cb4639f.png



结论:

1、NSObject 对象在内存中就是一个结构体(结构体优势:可以兼容不同种数据类型)。

4d1c5a48e54241beb008f2857a647843.png

2、Class:是指向结构体的指针。

3、指针占位大小:64位环境占8个字节(下面默认按64位系统计算),32位环境占4个字节。

4、而 NSObject_IMPL 只有一个成员对象,那么 isa 成员的地址就是结构体在内存中的地址,占8个字节。


NSObject *obj = [[NSObject alloc] init]; 代码解读:


// 1、创建好一个对象并给这个对象分配内存地址,假设地址为:0x100400110
[[NSObject alloc] init];
// 2、设置一个指针 *obj
NSObject *obj;
// 3、让 *obj 指针指向/存储这个对象的内存地址。
NSObject *obj = [[NSObject alloc] init];


图如下:

8bd77a9adbda47838106f3def6a8f897.png


三、NSObject 是有多少内存呢?


继续以上例子,已知有两个方法可以获取内存分配空间:


#import <objc/runtime.h>
#import <malloc/malloc.h>
// 获得NSObject实例对象的成员变量所占用的大小 >> 8
NSLog(@"%zd", class_getInstanceSize([NSObject class]));
// 获得obj指针所指向内存的大小 >> 16
NSLog(@"%zd", malloc_size((__bridge const void *)obj));


为什么一个返回8字节,另一个返回16字节。注意了,它们虽然都是获取占用内存大小。但是获取对象不一样。


class_getInstanceSize 方法获取的是实例对象的成员变量所占用的大小。

malloc_size 获取的是指针所指向的内存大小。

那为什么指针所指向的内存会更大呢?这时可以查看一下OC底层原码是如何分配内存的,如下:


 

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


可以看到,当分配内存小于16位时,直接设置16,这是他们底层的规定。


结论:

1、系统分配了16个字节给 NSObject 对象(通过malloc_size函数获得)

2、但 NSObject 对象内部只使用了8个字节的空间(64bit环境下,可以通过class_getInstanceSize函数获得)


四、Demo


InterView-NSObject的内存分析


相关文章
|
7月前
|
开发者
简述函数和框架的区别
简述函数和框架的区别
43 1
|
8月前
|
存储 C++
【C++】多态(重写)的实现过程及其原理【核心知识点精讲】(22)
【C++】多态(重写)的实现过程及其原理【核心知识点精讲】(22)
|
7月前
|
编译器 C++
c++primer plus 6 读书笔记 第十章 对象和类
c++primer plus 6 读书笔记 第十章 对象和类
|
3月前
|
存储 编译器 C语言
C++类与对象深度解析(一):从抽象到实践的全面入门指南
C++类与对象深度解析(一):从抽象到实践的全面入门指南
64 8
|
4月前
|
Java Python
全网最适合入门的面向对象编程教程:50 Python函数方法与接口-接口和抽象基类
【9月更文挑战第18天】在 Python 中,虽无明确的 `interface` 关键字,但可通过约定实现类似功能。接口主要规定了需实现的方法,不提供具体实现。抽象基类(ABC)则通过 `@abstractmethod` 装饰器定义抽象方法,子类必须实现这些方法。使用抽象基类可使继承结构更清晰、规范,并确保子类遵循指定的方法实现。然而,其使用应根据实际需求决定,避免过度设计导致代码复杂。
|
8月前
|
算法 Serverless 数据安全/隐私保护
【C++ 函数 基本教程 第三篇 】深度解析C++函数类型:探寻全局函数、成员函数与静态函数的奥秘
【C++ 函数 基本教程 第三篇 】深度解析C++函数类型:探寻全局函数、成员函数与静态函数的奥秘
501 1
|
Java
Java面向对象基础2——封装
简单来说,封装就是正确地设计对象的属性。要注意的是,对象代表什么,就封装对应的数据,并提供数据的对应行为
123 0
Java面向对象基础2——封装
|
Python
python类与对象入门笔记(基础概念扫盲,面向对象三大特征,多态性与鸭子类型,附demo)
python类与对象入门笔记(基础概念扫盲,面向对象三大特征,多态性与鸭子类型,附demo)
121 0
|
Java API
一网打尽“类”的初始化实例化知识点
之前说了类加载的过程,但是有的读者表示还是有些面试题还是答不来,所以今天就来总结下类加载、对象实例化方面的知识点/面试题,帮助大家加深印象。
151 0
一网打尽“类”的初始化实例化知识点
【Groovy】MOP 元对象协议与元编程 ( 方法合成引入 | 类内部获取 HandleMetaClass )
【Groovy】MOP 元对象协议与元编程 ( 方法合成引入 | 类内部获取 HandleMetaClass )
170 0
【Groovy】MOP 元对象协议与元编程 ( 方法合成引入 | 类内部获取 HandleMetaClass )