【iOS】alloc init new底层原理

目录

前言

alloc

alloc核心操作

cls->instanceSize(extraBytes)

calloc

obj->initInstanceIsa

init

类方法:

实例方法:

new


前言

笔者最近在进行对OC语言源码的学习,学习源码的过程中经常会出现一些从来没有遇见过的函数,因此很难把整个过程理解清楚,这篇博客先简单梳理一下我现在理解的 alloc & init & new 的实现过程以及内存对齐原理

alloc

首先从main函数中找到WGPerson类的alloc方法的实现:

在这个方法中,调用了_objc_rootAlloc,跳转到该函数的实现:

这个方法中又调用了callAlloc函数,同样跳转到该函数的实现:

这个函数是runtime中分配对象的核心方法之一,用于决定走哪条路径调用alloc或allocWithZone:

static ALWAYS_INLINE id
callAlloc(Class cls, bool checkNil, bool allocWithZone=false)

这两行是这个函数的定义部分,static说明这个函数只能在当前文件中使用,ALWAYS_INLINE表示这个函数应尽量内联,提高性能。

函数有三个参数,cls表示要分配内存的类,checkNil是一个bool变量,用于表明是否需要检查cls是否为nil,allocWithZone表示是否使用allocWithZone:方法。

函数内部实现里有三个判断,第一个判断是类是否为nil

#if __OBJC2__if (slowpath(checkNil && !cls)) return nil;

__OBJC2__表示仅在OC2.0环境下有效,<!--slowpath用于分支预测优化,提示这个条件大概率为假-->,checkNil && !cls 表示如果要求检查且cls是nil,就直接返回nil。

 if (fastpath(!cls->ISA()->hasCustomAWZ())) {return _objc_rootAllocWithZone(cls, nil);}
#endif

<!--fastpath与slowpath相对,也是用于分支预测优化的,它提示这个条件大概率为真。-->cls->ISA()用于获取类的元类,hasCustomAWZ()用于判断类是否重写了allocWithZone:这个方法,

进入条件语句后,_objc_rootAllocWithZone(cls, nil)是调用runtime的根分配方法 ,分配对象。

如果没有进入这条快路径,就只能走慢路径,发消息调用alloc/allocWithZone:

if (allocWithZone) {return ((id(*)(id, SEL, struct _NSZone *))objc_msgSend)(cls, @selector(allocWithZone:), nil);}

objc_msgSend:是runtime的消息发送函数,cls和@selector(allocWithZone:)是这个函数的两个参数,这条语句前面的部分((id(*)(id, SEL, struct _NSZone *))是在对objc_msgSend:函数进行类型转换,把函数转换成了返回一个id,接收两个参数:id,SEL的函数,这样来正确调用alloc方法。

这条语句就相当于给类对象cls发送alloc消息,从而创建一个该类的实例对象。

最后一种情况是当allocWithZone为假时,正常调用 alloc走常规路径,objc_msgSend: 发送 alloc消息,等效于[cls alloc]

在这里我们对自定义类进行观察,会在这几个分支中走到_objc_rootAllocWithZone,接着我们跳转到_objc_rootAllocWithZone的实现。

再继续跳转到_class_createInstanceFromZone的源码实现,这个部分是alloc源码的核心操作,实现主要分为三个部分:

  • cls->instanceSize:计算需要开辟的<!--内存空间大小-->

  • calloc:<!--申请内存-->,返回地址指针

  • obj->initInstanceIsa:将类与isa关联

static ALWAYS_INLINE id
_class_createInstanceFromZone(Class cls, size_t extraBytes, void *zone,int construct_flags = OBJECT_CONSTRUCT_NONE,bool cxxConstruct = true,size_t *outAllocatedSize = nil)
{ASSERT(cls->isRealized());
​// Read class's info bits all at once for performancebool hasCxxCtor = cxxConstruct && cls->hasCxxCtor();bool hasCxxDtor = cls->hasCxxDtor();bool fast = cls->canAllocNonpointer();size_t size;
​size = cls->instanceSize(extraBytes);if (outAllocatedSize) *outAllocatedSize = size;
​id obj;if (zone) {obj = (id)malloc_zone_calloc((malloc_zone_t *)zone, 1, size);} else {obj = (id)calloc(1, size);}if (slowpath(!obj)) {if (construct_flags & OBJECT_CONSTRUCT_CALL_BADALLOC) {return _objc_callBadAllocHandler(cls);}return nil;}
​if (!zone && fast) {obj->initInstanceIsa(cls, hasCxxDtor);} else {// Use raw pointer isa on the assumption that they might be// doing something weird with the zone or RR.obj->initIsa(cls);}
​if (fastpath(!hasCxxCtor)) {return obj;}
​construct_flags |= OBJECT_CONSTRUCT_FREE_ONFAILURE;return object_cxxConstructFromClass(obj, cls, construct_flags);
}

alloc核心操作

cls->instanceSize(extraBytes)

instanceSize(extraBytes)的实现如下:

instanceSize的调用过程如下:

这里自定义类最终走到fastInstanceSize

在fastInstanceSize中,最终调用的是align16这个函数,在834源码中使用是align16这个函数,在最新版xcode中会报红。

但是可以看到在906源码中其实也调用的是align16这个函数。这个函数的实现是一个16字节对齐算法

这段算法的过程就是将原始的内存 与size_t(15)相加,得到一个数,将 size_t(15) 即 15进行~(取反)操作,再将 这个数 与 15的取反结果 进行 &(与)操作,最后的结果为 16的倍数,即内存的大小是以16的倍数增加的

原理是因为内存必须比数据大,内存对齐的结果只能大于实际数据大小,而不能比它小。将数字加15,使数字大于等于比数据大小要大的最小的那个16的倍数,在和末四位为0的数字进行与操作,抹去后四位,去掉余数,变成16的倍数。

为什么需要内存对齐?

通常内存是由一个个字节组成的,cpu在存取数据时,并不是以字节为单位存储,而是以为单位存取,块的大小为内存存取力度。频繁存取字节未对齐的数据,会极大降低cpu的性能,所以可以通过减少存取次数降低cpu的开销

16字节对齐,是由于在一个对象中,第一个属性isa8字节,当然一个对象肯定还有其他属性,当无属性时,会预留8字节,即16字节对齐,如果不预留,相当于这个对象的isa和其他对象的isa紧挨着,容易造成访问混乱

16字节对齐后,可以加快CPU读取速度,同时使访问更安全,不会产生访问混乱的情况

calloc

calloc函数用于申请内存并返回地址指针

obj = (id)calloc(1, size);

这一行代码就是在用计算出来的size获取地址指针obj,此时地址还没有与传入的cls进行关联。

obj->initInstanceIsa

这一步是在将类与isa关联。内存申请好后,将传入的类与已经申请好的内存进行关联,而关联的方式就是isa指针。关联的流程如下:

在执行完initInstanceIsa后,内存便与类关联了起来。

综上,alloc的核心操作就是三步:计算内存,申请内存,内存与类关联。

init

init有两种,一种是类的init,一种是对象的。

类方法:

+ (id)init {return (id)self;
}

这里的init是一个构造方法 ,是通过工厂设计(工厂方法模式),主要是用于给用户提供构造方法入口。这里能使用id强转的原因,主要还是因为 内存字节对齐后,可以使用类型强转为你所需的类型

实例方法:

init实例方法会跳转到_objc_rootInit方法,来看看它的实现

可以发现,函数返回的是传入的self本身。

new

除了alloc与init,初始化还可以使用new方法

new其实就是调用了callAlloc函数(即alloc中分析的函数)以及init函数,因此就相当于[[ alloc] init]。

但是如果重写了init方法做一些自定义操作,这时会在这个方法中调用[super init],这时不建议使用new进行初始化。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/80359.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

QT窗口相关控件及其属性

widget&#xff0c;PushButton&#xff0c;lineEdit等都是基于QWidget延展出来的 并不是完整的窗口&#xff0c;而是作为窗口的一部分 真正的窗口是QMainWindow 菜单栏 Qt中的菜单栏是通过QMenuBar这个类来实现的&#xff0c;一个主窗口最多只有一个菜单栏&#xff0c;位于主…

day47—双指针-平方数之和(LeetCode-633)

题目描述 给定一个非负整数 c &#xff0c;你要判断是否存在两个整数 a 和 b&#xff0c;使得 a^2 b^2 c 。 示例 1&#xff1a; 输入&#xff1a;c 5 输出&#xff1a;true 解释&#xff1a;1 * 1 2 * 2 5示例 2&#xff1a; 输入&#xff1a;c 3 输出&#xff1a;f…

蓝桥杯 20. 压缩变换

压缩变换 原题目链接 题目描述 小明最近在研究压缩算法。他知道&#xff0c;压缩时如果能够使数值很小&#xff0c;就能通过熵编码得到较高的压缩比。然而&#xff0c;要使数值变小是一个挑战。 最近&#xff0c;小明需要压缩一些正整数序列&#xff0c;这些序列的特点是&a…

element-ui多个form同时验证,以及动态循环表单注意事项

多个form同时验证&#xff1a; validateForm(refs) {if (!refs) {return false}return new Promise((resolve, reject) > {refs.validate().then((valid) > {resolve(valid)}).catch((val) > {resolve(false)})}) }, async handleConfirm() {Promise.all([this.valid…

Spring Boot中自定义404异常处理问题学习笔记

1. 问题背景 在Spring Boot项目中&#xff0c;需要手动返回404异常给前端。为此&#xff0c;我创建了一个自定义的404异常类UnauthorizedAccessException&#xff0c;并在全局异常处理器GlobalExceptionHandler中处理该异常。然而&#xff0c;在使用Postman测试时&#xff0c;…

你学会了些什么220622?--搭建UI自动化

jenkins访问地址&#xff1a;http://192.168.82.129:8080/ 账号密码&#xff1a;admin/a123456a ***** 什么是UI自动化** 使用工具或者脚本对需要测试的软件的前端界面在预设的条件下&#xff0c;在已有的测试数据下运行系统或者应用程序&#xff0c;并获取其前端页面UI显示的…

【2025计算机网络-面试常问】http和https区别是什么,http的内容有哪些,https用的是对称加密还是非对称加密,流程是怎么样的

HTTP与HTTPS全面对比及HTTPS加密流程详解 一、HTTP与HTTPS核心区别 特性HTTPHTTPS协议基础明文传输HTTP SSL/TLS加密层默认端口80443加密方式无加密混合加密&#xff08;非对称对称&#xff09;证书要求不需要需要CA颁发的数字证书安全性易被窃听、篡改、冒充防窃听、防篡改…

JavaFX 第一篇 Hello World

1、简介 JavaFX 是一个用于构建客户端应用程序的 Java 库&#xff0c;作为 Java 标准库的一部分&#xff08;JDK 8 到 10&#xff09;&#xff0c;从 JDK 11 开始&#xff0c;JavaFX 将以独立模块发布&#xff0c;将不再包含在 JDK标准库中&#xff0c;他是 Java 应用程序开发的…

SQL实战:02之连续数问题求解

文章目录 概述题目:体育馆的人流量题解步骤一&#xff1a;构造出一个连续序列步骤二&#xff1a;找出符合条件的组的序号步骤三&#xff1a;fetch结果&#xff0c;使用内连接过滤出符合条件的记录。完整SQL 题目二&#xff1a;连续出现的数字题解步骤一&#xff1a;分区并构建连…

STM32 的 GPIO和中断

GPIO的简单介绍 内部结构 施密特触发器&#xff08;TTL肖特基触发器&#xff09; 的工作原理&#xff1a; 施密特触发电路&#xff08;简称&#xff09;是一种波形整形电路&#xff0c;当任何波形的信号进入电路时&#xff0c;输出在正、负饱和之间跳动&#xff0c;产生方波或…

Server - 优雅的配置服务器 Bash 环境(.bashrc)

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/147335592 免责声明&#xff1a;本文来源于个人知识与公开资料&#xff0c;仅用于学术交流&#xff0c;欢迎讨论&#xff0c;不支持转载。 登录服…

使用PyTorch实现图像增广与模型训练实战

本文通过完整代码示例演示如何利用PyTorch和torchvision实现常用图像增广方法&#xff0c;并在CIFAR-10数据集上训练ResNet-18模型。我们将从基础图像变换到复杂数据增强策略逐步讲解&#xff0c;最终实现一个完整的训练流程。 一、图像增广基础操作 1.1 准备工作 #matplotli…

解决Mac 安装 PyICU 依赖失败

失败日志&#xff1a; 解决办法 1、使用 homebrew 安装相关依赖 brew install icu4c 安装完成后&#xff0c;设置环境变量 echo export PATH"/opt/homebrew/opt/icu4c77/bin:$PATH" >> ~/.zshrcecho export PATH"/opt/homebrew/opt/icu4c77/sbin:$PATH…

Springboot后端查询参数接收

1.实现方式 假设前端发送的接口&#xff1a; /users?nameJohn&age30 后端怎么接收里面的name和age呢&#xff1f;以及再发别的参数后端怎么接收呢&#xff1f; 1.比较简单的方式 当控制器方法的参数类型是简单类型&#xff08;如 String、Integer、Long 等&#xff09…

桌面应用中VUE使用新浏览器窗口打开页面

1、浏览器应用忽略此方式&#xff0c;可任意方式打开。针对桌面应用设置 newWindowClick(){try {this.fileUrl "";this.params.year ""this.params.date ""axios({method: post,url: /url/pdf/preview,data: this.params,}).then(res> {t…

华为手机怎么进行音频降噪?音频降噪技巧分享:提升听觉体验

在当今数字化时代&#xff0c;音频质量对于提升用户体验至关重要&#xff0c;无论是在通话、视频录制还是音频文件播放中&#xff0c;清晰的音频都能带来更佳的听觉享受。 而华为手机凭借其强大的音频处理技术&#xff0c;为用户提供了多种音频降噪功能&#xff0c;帮助用户在…

【数据可视化-22】脱发因素探索的可视化分析

🧑 博主简介:曾任某智慧城市类企业算法总监,目前在美国市场的物流公司从事高级算法工程师一职,深耕人工智能领域,精通python数据挖掘、可视化、机器学习等,发表过AI相关的专利并多次在AI类比赛中获奖。CSDN人工智能领域的优质创作者,提供AI相关的技术咨询、项目开发和个…

青少年编程与数学 02-018 C++数据结构与算法 06课题、树

青少年编程与数学 02-018 C数据结构与算法 06课题、树 一、树(Tree)1. 树的定义2. 树的基本术语3. 常见的树类型4. 树的主要操作5. 树的应用 二、二叉树(Binary Tree)1. 二叉树的定义2. 二叉树的基本术语3. 二叉树的常见类型4. 二叉树的主要操作5. 二叉树的实现代码说明输出示例…

【论文阅读】Visual Instruction Tuning

文章目录 导言1、论文简介2、论文主要方法3、论文针对的问题4、论文创新点总结 导言 本论文介绍了一个新兴的多模态模型——LLaVA&#xff08;Large Language and Vision Assistant&#xff09;&#xff0c;旨在通过指令调优提升大型语言模型&#xff08;LLM&#xff09;在视觉…

【学习笔记】Cadence电子设计全流程(三)Capture CIS 原理图绘制(下)

【学习笔记】Cadence电子设计全流程&#xff08;三&#xff09;Capture CIS 原理图绘制&#xff08;下&#xff09; 3.16 原理图中元件的编辑与更新3.17 原理图元件跳转与查找3.18 原理图常见错误设置于编译检查3.19 低版本原理图文件输出3.20 原理图文件的锁定与解锁3.21 Orca…