【iOS】OC高级编程 iOS多线程与内存管理阅读笔记——自动引用计数(一)

news2025/5/26 0:55:55

自动引用计数

  • 前言
  • alloc/retain/release/dealloc实现
    • 苹果的实现
  • autorelease
  • autorelease实现
    • 苹果的实现
  • 总结

前言

此前,写过一遍对自动引用计数的简单学习,因此掠过其中相同的部分:引用计数初步学习


alloc/retain/release/dealloc实现

由于NSObject类的源码不公开,我们通过开源软件GNUstep来学习相关内容。

GNUstep是Cocoa框架的互换框架,虽然并不是与苹果Cocoa的实现方式完全相同,但是从使用者的角度来看二者的行为和实现方式是一样的,理解了GNUstep的源代码也相当于理解了苹果的Cocoa实现。

先来看alloc类方法。

id obj = [NSObject alloc];

上述调用NSObject类的alloc类方法在NSObject.m的源代码中的实现如下:

+ (id) alloc
{
    return [self allocWithZone: NSDefaultMallocZone()];
}

+ (id) allocWithZone: (NSZone*) z
{
    return NSAllocateObject(self, 0, z);
}

上述代码使用了在allocWithZone类方法中,使用NSAllocateOject函数来分配对象。下面是这个函数的代码:

struct obj_layout {
    NSUInteger retained;
};

inline id
NSAllocateObject(Class aClass, NSUInteger extraBytes, NSZone *zone)
{
    int size = 计算容纳对象所需内存大小;
    id new = NSZoneMalloc(zone, size);
    memset(new, 0, size);
    new = (id) &((struct obj_layout *) new)[1];
}

NSAllocateOject函数通过调用NSZoneMalloc函数来进行分配存放对象所需的内存空间,之后将该内存空间置0,最后返回作为对象而使用的指针。

NSZone是为了防止内存碎片化而引入的结构。对内存分配的区域本身进行多重化管理,根据使用对象的目的和大小分配内存,从而提高内存管理的效率。
请添加图片描述
因此简化NSZone方法后的alloc源代码如下:

// 定义 obj_layout 结构体,用于存储引用计数
struct obj_layout {
    NSUInteger retained;
};

// 实现 alloc 方法
+ (id)alloc {
    // 计算对象所需的内存大小
    int size = sizeof(struct obj_layout) + class_getInstanceSize([self class]);
    
    // 分配内存并初始化为 0
    struct obj_layout *p = (struct obj_layout *)calloc(1, size);
    
    // 返回指向对象内存的指针(跳过 obj_layout)
    return (id)(p + 1);
}

alloc类方法使用obj_layout来保存对象的引用计数,记录在retain字段中,并将其写入对象内存头部,该对象全部置0后返回。alloc后返回的对象如图所示:

请添加图片描述
alloc后对象引用计数加一。下面是GNUstep源码:

-(NSUInterger) retainCount
{
	return NSextraRefCount (self) + 1;
}
inline NSUInteger
NSExtraRefCount(id anObject)
{
	return ((struct obj_layout *)anObject) [-1].retained;
}

这里需要由对象寻址找到对象内存头部,访问其中的retained变量。
请添加图片描述
retain方法使retained变量加1,release方法使retained变量减1。
下面是retain和release的源码:
retain源码:
请添加图片描述
retain的实例方法中是调用NSIncrementExtraRefCount函数,该函数的作用是使retained加1。并且为该变量超出最大值做出处理。

release源码:请添加图片描述

请添加图片描述
release方法先调用NSDecrementExtraRefCountWasZero函数,该函数的作用是让retained一直减到0。减到0后调用dealloc方法。废弃该对象。

上述代码仅废弃由alloc分配的内存块。

苹果的实现

使用NSObject类的alloc方法时,调用以下方法和函数

  • +alloc
  • +allocWithZone
  • class_createInstance
  • calloc

这个调用过程与前文所讲GNUstep相似,先调用allocWithZone方法,在调用class_createInstance函数,最后通过调用calloc来分配内存块。

接下来看retainCount/retain/release实例方法如何实现:
请添加图片描述
每个方法都调用了同一个函数_CFDoExternRefOperation该函数的前缀“CF“表明,它们包含于Core Foundation框架源代码中。我们来看其简化后的源码:

int __CFDoExternRefOperation(uintptr_t op, id obj) {
    CFBasicHashRef table = 取得对象对应的散列表(obj);
    int count;
    
    switch(op) {
    case OPERATION_retainCount:
      count = CFBasicHashGetCountOfKey(table, obj);
      return count;
    case OPERATION_retain:
      CFBasicHashAddValue(table, obj);
      return obj;
    case OPERATION_release:
      count = CFBasicHashRemoveValue(table, obj);
      return 0 == count;
    }
}

该函数按retainCount/retain/release操作进行分发,调用不同的函数。他们的实例方法可能如下所示:

- (NSUInteger)retainCount {
    return (NSUInteger)__CFDoExternRefOperation(OPERATION_retain, self);
}- (id)retain {
    return (id)__CFDoExternRefOperation(OPERATION_retain, self);
}- (void)release {
    return __CFDoExternRefOperation(OPERATION_release, self);
}

可以从__CFDoExternRefOperation函数以及由此函数调用的各个函数名中看出,苹果的实现大概采用散列表(引用计数表)来管理引用计数。如图所示:
请添加图片描述
GNUstep和苹果在实现引用计数的保存上有所不同

  • GNUstep将引用计数保存在对象占用内存头部的变量

少量代码即可完成,能够统一管理引用计数内存块与对象用内存块

  • 苹果保存在引用计数表中记录

对象用内存块的分配无需考虑内存块头部。
引用计数表各记录中存有内存块地址,可从各个记录中追溯到各对象的内存块。
这一点在调试故障时非常有效。即使出故障导致对象占用的内存块损坏,但只要引用计数表没有破坏,就能够确认各块内存块的位置。

请添加图片描述


autorelease

autorelease是自动释放,虽然看上去像ARC,但实际上更类似于C语言中的自动变量(局部变量)的特性。

autorelease会像C语言的局部变量那样对待对象实例。当超出其作用域时,对象的release实例方法才被调用。但是不同的是,autorelease可以设定变量的作用域。具体使用方法如下:

  1. 生成并持有NSAutoreleasePool对象
  2. 调用已分配对象的autorelease方法
  3. 废弃NSAutoreleasePool方法

NSAutoreleasePool对象的生存周期相当于C语言变量的作用域。对于所有调用过autorelease实例方法的对象,在废弃NSAutoreleasePool对象时,都将调用release实例方法。如图所示:

请添加图片描述
源代码如下:

NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];

id obj = [[NSObject alloc] init];

[obj autoreless];

[pool drain];

其中,[pool drain]方法等同于[obj release]

在Cocoa框架中,相当于程序主循环的NSRunLoop或者在其他程序可运行的地方,对NSAutoreleasePool对象进行生成,持有,和废弃处理。
请添加图片描述
但是,只要不废弃NSAutoreleasePool对象,那么生成的对象就不能释放,因此会产生内存不足的现象。我们只需要自定义一个pool,在最后进行[pool drain]

for (int i = 0; i < image.count; i++) {
	NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]];
/*
	产生autorelease对象
*/
[pool drain]
}
}

许多类方法也会返回autorelease对象,如NSMutableArray类的arrayWithCapacity类方法。


autorelease实现

我们先来看一下autorelease源码

-(id) autorelease
{
	[NSAutoreleasePool addObject:self];
}

该方法的本质就是调用NSAutoreleasePool对象的addObject类方法。


IMP Caching
当我们调用一个方法时,OC通过消息传递机制实现 :

  1. 查找方法名(selector)。
  2. 找到方法的实现(IMP,即函数指针)。
  3. 执行方法。

这个过程非常灵活,但是如果需要频繁的调用方法,如上文的autorelease方法,那么则会带来一定的性能开销。

为了减少这种开销,我们则采用IMP Caching技术:

  • 在程序初始化时,预先查找并缓存方法的实现(IMP)。
  • 在后续调用时,直接使用缓存的 IMP,避免重复查找。

通常情况下,IMP Caching的速度是普通方法的两倍。尤其是在频繁调用方法时。


现在我们看NSAutoreleasePool类的实现。由于该类的源代码比较复杂,因此我们假想一个简化的源代码学习:

+ (void) addObject:(id) anObj
{
	NSAutoreleasePool *pool = 取得正在使用的NSAutoreleasePool对象;
	if(pool != nil) {
		[pool addObject:anObj];
	} else {
		NSLog(@"NSAutoreleasePool对象非存在状态下调用autorelease");
	}
}

addObject类方法调用正在使用的NSAutoreleasePool对象的addObject实例方法。

NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
id obj = [[NSObject alloc] init];
[obj autorelease];

上述是被赋予pool变量的即为正在使用的NSAutoreleasePool对象。

如果嵌套生成或持有的NSAutoreleasePool对象,则会使用最内侧的对象。

NSAutoreleasePool *pool0 = [[NSAutoreleasePool alloc] init];
NSAutoreleasePool *pool1 = [[NSAutoreleasePool alloc] init];
NSAutoreleasePool *pool2 = [[NSAutoreleasePool alloc] init];

id obj = [[NSObject alloc] init];
[obj autorelease]; // obj 被添加到 pool2 中

[pool2 drain]; // 释放 pool2 中的对象
[pool1 drain]; // 释放 pool1 中的对象
[pool0 drain]; // 释放 pool0 中的对象

上述代码中,obj会被添加到最内层的pool2中。当调用drain时,先释放pool2,再释放pool1和pool0。

我们继续看addObject实例方法的实现。

-(void) addObject:(id) anObj
{
	[array addObject:anObj];
}

GNUstep实现使用的是连接列表,这同在NSMutableArray对象中追加对象参数是一样的。

如果调用NSObject类的autorelease实例方法,该对象被追加到正在使用的NSAutoreleasePool对象中的数组里。

下面我们看drain实例方法

- (void)drain {
    [self dealloc];
}

- (void)dealloc {
    [self emptyPool];
    [array release];
}

- (void)emptyPool {
    for (id obj in array) {
        [obj release];
    }
}

该方法会释放pool中所有的对象。


苹果的实现

class AutoreleasePoolPage {
public:
    static inline void *push() {
		相当于生成或持有NSAutoreleasePool类对象;
    }

    // 释放自动释放池
    static inline void pop(void *token) {
		相当于废弃NSAutoreleasePool类对象;
		releaseAll();
    }

	static inline id autorelease (id obj)
	{
		相当于NSAutoreleasePool类的addObject类方法
		AutoreleasePoolPage *autoreleasePoolPage = 取得正在使用的AutoreleasePoolPage实例;
		autoreleasePoolPage->add(obj);
	}

    // 将对象添加到内部数组
    id *add(id obj) {
	
    }

    // 释放内部数组中的所有对象
    void releaseAll() {
        // 遍历内部数组,调用每个对象的 release 方法
        for (id obj : internalArray) {
            [obj release];
        }
        // 清空数组
        internalArray.clear();
    }

	// 创建一个新的自动释放池
void *objc_autoreleasePoolPush(void) {
    return AutoreleasePoolPage::push();
}

// 释放自动释放池
void objc_autoreleasePoolPop(void *ctxt) {
    AutoreleasePoolPage::pop(ctxt);
}

// 将对象添加到自动释放池
id objc_autorelease(id obj) {
    return AutoreleasePoolPage::autorelease(obj);
}

我们还可以使用NSAutoreleasePool类中的调试用非公开类方法showPools来确认已被autorelease的对象的状况。

autorelease NSAutoreleasePool对象会发生异常。

总结

对引用计数相关方法的实现原理进行简单了解。对比GNUstep和苹果对统一操作不同实现方法优劣的好坏。如引用计数的保存。

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

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

相关文章

Python爬虫第15节-2025今日头条街拍美图抓取实战

目录 一、项目背景与概述 二、环境准备与工具配置 2.1 开发环境要求 2.2 辅助工具配置 三、详细抓取流程解析 3.1 页面加载机制分析 3.2 关键请求识别技巧 3.3 参数规律深度分析 四、爬虫代码实现 五、实现关键 六、法律与道德规范 一、项目概述 在当今互联网时代&a…

智慧城市像一张无形大网,如何紧密连接你我他?

智慧城市作为复杂巨系统&#xff0c;其核心在于通过技术创新构建无缝连接的网络&#xff0c;使物理空间与数字空间深度融合。这张"无形大网"由物联网感知层、城市数据中台、人工智能中枢、数字服务入口和安全信任机制五大支柱编织而成&#xff0c;正在重塑城市运行规…

网络安全·第四天·扫描工具Nmap的运用

今天我们要介绍网络安全中常用的一种扫描工具Nmap&#xff0c;它被设计用来快速扫描大型网络&#xff0c;主要功能包括主机探测、端口扫描以及版本检测&#xff0c;小编将在下文详细介绍Nmap相应的命令。 Nmap的下载安装地址为&#xff1a;Nmap: the Network Mapper - Free Se…

黑龙江 GPU 服务器租用:开启高效计算新征程

随着人工智能、深度学习、大数据分析等技术的广泛应用&#xff0c;对强大计算能力的需求日益迫切。GPU 服务器作为能够提供卓越并行计算能力的关键设备&#xff0c;在这一进程中发挥着至关重要的作用。对于黑龙江地区的企业、科研机构和开发者而言&#xff0c;选择合适的 GPU 服…

SparseDrive---论文阅读

纯视觉下的稀疏场景表示 算法动机&开创性思路 算法动机&#xff1a; 依赖于计算成本高昂的鸟瞰图&#xff08;BEV&#xff09;特征表示。预测和规划的设计过于直接&#xff0c;没有充分利用周围代理和自我车辆之间的高阶和双向交互。场景信息是在agent周围提取&#xff…

Unchained 内容全面上链,携手 Walrus 迈入去中心化媒体新时代

加密新闻媒体 Unchained — — 业内最受信赖的声音之一 — — 现已选择 Walrus 作为其去中心化存储解决方案&#xff0c;正式将其所有媒体内容&#xff08;文章、播客和视频&#xff09;上链存储。Walrus 将替代 Unchained 现有的中心化存储架构&#xff0c;接管其全部历史内容…

确保连接器后壳高性能互连的完整性

本文探讨了现代后壳技术如何促进高性能互连的电气和机械完整性&#xff0c;以及在规范阶段需要考虑的一些关键因素。 当今的航空航天、国防和医疗应用要求连接器能够提供高速和紧凑的互连&#xff0c;能够承受振动和冲击&#xff0c;并保持对电磁和射频干扰 &#xff08;EMI/R…

C++学习Day0:c++简介

目录 一、.C语言的发展史二、C特点三、面向对象的重要术语四、面向过程和面向对象的区别&#xff1f;五、开发环境&#xff1a;六、创建文件步骤&#xff1a;1.点击新建项目2.在弹出的开始栏中按如下操作3.在.pro文件中添加&#xff08;重要&#xff01;&#xff01;&#xff0…

从零开始构建 Ollama + MCP 服务器

Model Context Protocol&#xff08;模型上下文协议&#xff09;在过去几个月里已经霸占了大家的视野&#xff0c;出现了许多酷炫的集成示例。我坚信它会成为一种标准&#xff0c;因为它正在定义工具与代理或软件与 AI 模型之间如何集成的新方式。 我决定尝试将 Ollama 中的一…

MATLAB学习笔记(二) 控制工程会用到的

MATLAB中 控制工程会用到的 基础传递函数表达传递函数 零极点式 状态空间表达式 相互转化画响应图线根轨迹Nyquist图和bode图现控部分求约旦判能控能观极点配置和状态观测 基础 传递函数表达 % 拉普拉斯变换 syms t s a f exp(a*t) %e的a次方 l laplace(f) …

C++ 线程间通信开发从入门到精通实战

C 线程间通信开发从入门到精通实战 在现代软件开发中&#xff0c;多线程程序已成为提升应用性能、实现并行处理的重要手段。随着多核处理器的普及和复杂应用需求的增加&#xff0c;C作为一门高性能的编程语言&#xff0c;在多线程开发中扮演着不可或缺的角色。然而&#xff0c…

Redis原理与Windows环境部署实战指南:助力测试工程师优化Celery调试

引言 在分布式系统测试中&#xff0c;Celery作为异步任务队列常被用于模拟高并发场景。而Redis作为其核心消息代理&#xff0c;其性能和稳定性直接影响测试结果。本文将深入解析Redis的核心原理&#xff0c;主要讲解Windows环境部署redis&#xff0c;为测试工程师提供一套完整…

Go语言入门到入土——一、安装和Hello World

Go语言入门到精通——安装和Hello World 文章目录 Go语言入门到精通——安装和Hello World下载并安装让Go跑起来为你的代码启动依赖跟踪调用外部包总结 下载并安装 下载地址&#xff1a;https://go.dev/dl/ 下载后傻瓜式安装 查看是否安装完成 go version让Go跑起来 创建一个…

opencv函数展示

一、图像基础 I/O 与显示 1.cv2.imread() 2.cv2.imshow() 3. cv2.waitKey() 4. cv2.imwrite() 5. cv2.selectROI() 6. cv2.VideoCapture() 二、颜色空间与转换 1. cv2.cvtColor() 2. cv2.split() 三、阈值处理 1. cv2.threshold() 2. 特殊阈值方法

【vue3】vue3+express实现图片/pdf等资源文件的下载

文件资源的下载&#xff0c;是我们业务开发中常见的需求。作为前端开发&#xff0c;学习下如何自己使用node的express框架来实现资源的下载操作。 实现效果 代码实现 前端 1.封装的请求后端下载接口的方法,需求配置aixos的请求参数里面的返回数据类型为blob // 下载 export…

如何在 Kali 上解决使用 evil-winrm 时 Ruby Reline 的 quoting_detection_proc 警告

在使用 Kali Linux 运行 Ruby 工具&#xff08;例如 evil-winrm&#xff09;时&#xff0c;你可能会遇到以下警告&#xff1a; Warning: Remote path completions is disabled due to ruby limitation: undefined method quoting_detection_proc for module Reline这个警告会导…

从零到一:网站设计新手如何快速上手?

从零到一&#xff1a;网站设计新手如何快速上手&#xff1f; 在当今数字化时代&#xff0c;网站已成为企业、个人展示信息、提供服务的重要窗口。对于想要涉足网站设计领域的新手而言&#xff0c;如何快速上手并掌握必要的技能成为首要任务。本文将从基础知识、软件工具、设计…

面向初学者的JMeter实战手册:从环境搭建到组件解析

&#x1f31f; ​大家好&#xff0c;我是摘星&#xff01;​ &#x1f31f; 今天为大家带来的是面向初学者的JMeter实战手册&#xff1a;从环境搭建到组件解析&#xff0c;废话不多说&#xff0c;让我们直接开始~ 目录 1. JMeter简介 2. JMeter安装与配置 2.1. 安装 2.2.…

工资管理系统的主要功能有哪些

工资管理系统通过自动化薪资计算、税务处理、员工数据管理、报表生成等功能&#xff0c;极大地提升了薪资发放的效率和准确性。在传统的人工薪资管理中&#xff0c;HR人员需要手动计算每位员工的薪资&#xff0c;并确保符合税务要求&#xff0c;极易出错且耗时。而现代工资管理…

子函数嵌套的意义——以“颜色排序”为例(Python)

多一层缩进精减参数传递&#xff0c;参数少平铺书代码写更佳。 笔记模板由python脚本于2025-04-16 11:52:53创建&#xff0c;本篇笔记适合喜欢子函数嵌套结构代码形式的coder翻阅。 【学习的细节是欢悦的历程】 博客的核心价值&#xff1a;在于输出思考与经验&#xff0c;而不仅…