【iOS】ARC学习

文章目录

  • 前言
  • 一、autorelease实现
  • 二、苹果的实现
  • 三、内存管理的思考方式
    • __strong修饰符
      • 取得非自己生成并持有的对象
      • __strong 修饰符的变量之间可以相互赋值
      • 类的成员变量也可以使用strong修饰
    • __weak修饰符
      • 循环引用
    • __unsafe_unretained修饰符
      • 什么时候使用__unsafe_unretained
    • __autoreleasing修饰符
      • 访问附有__weak 修饰符的变量时,实际上必定要访问注册到autoreleasepool的对象
  • 四、ARC规则
    • 不要显式调用dealloc
  • 五、属性


前言

在学习ARC之前,先来复习一下内存管理以及autorelease的实现

一、autorelease实现

先来看一下GNUstep源代码:
在这里插入图片描述
autorelease其本质就是调用NSAutoreleasePool 对象的addObject 类方法,就是将对象加到自动释放池中
在这里插入图片描述
接下来再看一下废弃自动释放池的一些功能函数
在这里插入图片描述

二、苹果的实现

可使用showPools输出现在的NSAutoreleasePool的状况输出到控制台

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

    [obj autorelease];
    [obj2 autorelease];
    [obj3 autorelease];

    _objc_autoreleasePoolPrint();
    [pool drain];

在这里插入图片描述

三、内存管理的思考方式

引用计数式内存管理的思考方式就是思考ARC所引起的变化

ARC有效时,id 类型和对象类型同C语言其他类型不同,其类型上必须附加所有权修饰符。 所有权修饰符一共有4 种

(ARC环境下特有)

  • __strong修饰符
  • __weak修饰符
  • __unsafe_unretained修饰符
  • __autoreleasing修饰符

__strong修饰符

_ strong修饰符是id类型和对象类型默认的所有权修饰符
也就是说id obj = [[NSObject alloc] init]; = id __strong obj = [[NSObject alloc] init];

再来看下面这段代码

//ARC有效
{
	id __strong obj = [[NSObject alloc] init]
}

此源代码指定了变量的作用域,当obj超出其变量作用域时, obj会被废弃,同时自动释放其被赋予的对象([[NSObject alloc] init])

而在MRC中,等效的代码为

//ARC无效
{
	id obj = [[NSObject alloc] init]
	[obj release];
}

因为ARC无效的时候,obj超出变量作用域时,变量并不会被自动废弃,对象也会仍然存在,需要我们手动减少对象的引用计数[obj release]去销毁对象

取得非自己生成并持有的对象

具体如下:
在这里插入图片描述
这里需要注意的一点是此处obj确实持有了对象,并且对象的引用计数为1,但是在目前版本的Xcode的MRC环境中,

    {
        id obj = [NSMutableArray array];
        NSLog(@"%lu", [obj retainCount]);
    }

在MRC环境下输出的值应该为0,因为array方法表明取得非自己生成并持有的对象,也就是说obj并不持有对象,但是输出如下:
在这里插入图片描述
我们来解释一下输出为1的原因:

当我们调用 retainCount 方法时,对于从自动释放池获取的对象,它会临时retained一次,以防止对象被过早释放而导致访问过期数据。

1.array方法创建了一个对象,并将其加到自动释放池中,此时retain count为0
2.obj指向自动释放池中的那个对象,并没有对对象进行retain操作,只是持有了一个指向他的指针
3.调用 [obj retainCount] 时 ,a.编译器会临时保留(retain)对象 b.获得并输出其retain count值 c.释放对象

所以尽管对象最初的 retain count 为 0,但由于 retainCount 方法的实现机制,它会临时保留对象来避免崩溃,这导致我们看到的输出 retain count 为 1。

__strong 修饰符的变量之间可以相互赋值

   id __strong obj0 = [[NSObject alloc] init];//对象A
   id __strong obj1 = [[NSObject alloc] init];//对象B
   id __strong obj2 = nil;
    obj0 = obj1;//obj0持有有obj1赋值的对象B的强引用,obj0被赋值。所以原先持有的a的强引用失效,此时b的强引用变量为obj1,obj0

由此__strong 修饰符的变量,不仅只在变量作用域中,在赋值上也能够正确地管理其对象的所有者。

类的成员变量也可以使用strong修饰

在这里插入图片描述
在这里插入图片描述
重点是当Test对象释放时,Test对象的obj_成员变量也会随之被释放

__weak修饰符

使用weak可以使我们取得对象但是并不持有对象
下面有一个代码例子进行解释

    id a = [[NSObject alloc] init];
    id __weak b = a;
    id c = a;
    NSLog(@"%lu", CFGetRetainCount((__bridge  CFTypeRef)a));
    NSLog(@"%lu", CFGetRetainCount((__bridge  CFTypeRef)c));
    a = nil;
    NSLog(@"%lu", CFGetRetainCount((__bridge  CFTypeRef)c));
    NSLog(@"%@", b);
    NSLog(@"%@", c);
    c = nil;
    NSLog(@"%@", b);
    NSLog(@"%@", c);

在这里插入图片描述
可以看到在ARC环境下a,b,c都指向了对象,但是引用计数只有2,这是因为weak是指向对象的指针但并不持有对象,并不会使引用计数加1

当我们将weak修饰符改为strong时,就会出现如下结果

    id a = [[NSObject alloc] init];
    id __strong b = a;
    id c = a;
    NSLog(@"%lu", CFGetRetainCount((__bridge  CFTypeRef)a));
    NSLog(@"%lu", CFGetRetainCount((__bridge  CFTypeRef)c));
    a = nil;
    NSLog(@"%lu", CFGetRetainCount((__bridge  CFTypeRef)c));
    NSLog(@"%@", b);
    NSLog(@"%@", c);
    c = nil;
    NSLog(@"%@", b);
    NSLog(@"%@", c);

在这里插入图片描述

循环引用

内存管理中会发生循环引用的问题,此时就需要用到__weak修饰符
在这里插入图片描述

正确的内存释放过程
首先B对象是A对象的一个属性,也就是A持有B,现在要释放掉A,需要给A发送一个release消息,这时A的引用计数变为0,就要走delloc方法,delloc方法会对A所持有的全部对象发送release消息,当然也包括B,也就是对B进行release,此时B的引用计数也变为0,然后执行delloc,最后A与B都被释放掉了

- (void)dealloc {
    [_b release]; // 释放持有的 B 实例
    _b = nil;
    [super dealloc];
}

在这里插入图片描述
循环引用的产生:
解释:对象之间互相持有,形成闭环,导致谁也无法被正确释放
循环引用容易发生内存泄漏。所谓内存泄漏就是应当废弃的对象在超出其生存周期后继续存在。
在这里插入图片描述

过程:

  • a.例如我们现在想让A释放,也就是让B给A发送release消息,此时B的属性强持有A,所以需要B在delloc方法中对A进行release
  • b.我们想要让B执行delloc,就需要就持有B的A对象发送release消息给B
  • c.想要A发送release消息给B,就需要A执行delloc方法
  • d.想要A执行delloc方法就需要持有A的B对象发送release消息

如此循环往复,对象之间都在等对方给自己发送release消息,导致谁也无法执行,如此往复便造成了循环引用

当然循环引用并不只出现在变量中,还出现在协议与block中,后面在学习的过程中会专门写博客记录

接下来我们谈论一下如何解决这类问题

因为我们知道weak可以使变量取得但并不持有对象,也就是说不会增加对象的引用计数,我们将对象中的属性用weak修饰符修饰就可以解决这个问题
在这里插入图片描述

在这里插入图片描述

使用weak时因为变量不持有对象,因此不会造成相互引用,当对象释放后weak变量会自动置为nil,也避免了野指针的情况

#pragma mark 持有对象的弱引用 MRC 在MRC下,没有__weak这样的自动nil置化特性    使用weak持有某对象弱引用时,对象被废弃,弱引用变量自动只为nil
    id __weak obj1 = nil;
    {
        id __strong obj0 = [[NSObject alloc] init];
        obj1 = obj0;;
        NSLog(@"%@", obj1);
    }
    NSLog(@"%@", obj1);

在这里插入图片描述

__unsafe_unretained修饰符

__unsafe_unretained是一个不安全的所有权修饰符,在MRC下使用来避免循环引用
在这里插入图片描述

但是与weak相比,其会产生悬垂指针

因此我们在使用**__unsafe_unretained必须保证对象存在**

什么时候使用__unsafe_unretained

  • 其与weak相比,可能会有一些更好的性能,追求极致性能便可以使用__unsafe_unretained修饰
  • 以及在早版本的iOS中需要使用__unsafe_unretained来代替__weak.
  • 比如我们在访问单例或者全局变量时就可以使用这个修饰

__autoreleasing修饰符

ARC中不能使用autorelease方法以及NSAutoreleasePool类,但是实际上ARC有效时autorelease功能还是有作用的

  1. 指定“@autoreleasepol 块”来替代“NSAutoreleasePool 类对象生成、持有以及废弃”这一范围
@autoreleasepool{
id __autoreleasing obj = [[NSObject alloc] init];
}
  1. 为对象附加__autoreleasing修饰符代替autorelease方法,等价于在ARC无效时调用autorelease即将对象注册到autoreleasepool
    在这里插入图片描述
    但是我们一般不会显式地添加__autoreleasing,因为编译器会检查方法名是否以alloc/new/ copy/mutableCopy 开始,如果不是则自动将返回值的对象注册到autoreleasepool。
    例如:
@autoreleasepool f
id __strong obj = [NSMutableArray array];
}

访问附有__weak 修饰符的变量时,实际上必定要访问注册到autoreleasepool的对象

因为weak修饰符只持有对象的弱引用,因此访问引用对象时对象可能被遗弃。所以我们将对象注册到autoreleasepool可以确保对象存在
当将对象注册在autoreleasepool中,autoreleasepool会临时保留这个对象,直到作用域结束

另外在书上讲id *obj = id __autoreleasing *obj,以此类推NSObject **obj便成为了NSObject * _autoreleasing *obj,这里我们需要知道NSObject *__autoreleasing t1与NSObject __autoreleasing *t1有本质的不同:

  • 前者指向对象的对象会在被赋值时加入到自动释放池
  • 后者常在NSError错误处理中见到

在自动引用计数(ARC)管理的 Objective-C 环境中,当你使用双重指针(比如 NSError **error)作为方法参数时,ARC 会假定这个指针指向的对象是 __autoreleasing

总结来说,NSObject *__autoreleasing t2 是一个自动释放的对象指针,而 NSObject
__autoreleasing *t1 是指向一个自动释放对象指针的指针。

我们以一个代码例子来实验一下

@interface ViewController : UIViewController
@property (nonatomic, weak)NSObject *Obj1;
@property (nonatomic, weak)NSObject *Obj2;

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    [self test1];


}

- (void)test1 {
    NSLog(@"%@   %@", self.Obj1, self.Obj2);
    [self test2];
    NSLog(@"%@   %@", self.Obj1, self.Obj2);

}

- (void)test2 {
    NSObject *t1 = [[NSObject alloc] init];
    NSObject *__autoreleasing t2 = [[NSObject alloc] init];
    self.Obj1 = t1;
    self.Obj2 = t2;
    NSLog(@"%@   %@", self.Obj1, self.Obj2);

}

在这里插入图片描述
这段代码中因为test2中,t1超出变量作用域,同时self.obj1是被weak修饰的,并不持有对象,对象超出作用域自动销毁,因此第三行输出null。
而t2由于使用了 __autoreleasing,它的生命周期被延长到当前的自动释放池结束

它的生命周期被延长到当前的自动释放池结束"这句话的含义是:

  • 使用__autoreleasing修饰的对象不会在创建它的作用域(通常是一个函数或@autoreleasepool块)结束时被立即释放。
  • 相反,这个对象会被自动添加到当前的Autorelease Pool中,延长了它的生命周期。
  • 直到当前的Autorelease Pool被销毁时,这个对象才会被最终释放。
@interface ViewController : UIViewController
@property (nonatomic, strong)NSObject *Obj1;
@property (nonatomic, strong)NSObject *Obj2;

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    [self test1];


}

- (void)test1 {
    NSLog(@"%@   %@", self.Obj1, self.Obj2);
    [self test2];
    NSLog(@"%@   %@", self.Obj1, self.Obj2);

}

- (void)test2 {
    NSObject *t1 = [[NSObject alloc] init];
    NSObject *__autoreleasing t2 = [[NSObject alloc] init];
    self.Obj1 = t1;
    self.Obj2 = t2;
    NSLog(@"%@   %@", self.Obj1, self.Obj2);

}

在这里插入图片描述
而这段代码中obj用了strong修饰,即使t1,t2作为局部变量超出了作用域,但是self.Obj1仍然持有这个对象,因此这个对象并不会被销毁,因此其仍然存在

四、ARC规则

  • 不能使用retain/release/retainCount/autorelease
  • 不能使用NSAllocateObject/NSDeallocateObject
  • 必须遵守内存管理的方法名规则
  • 不要显式调用dealloc
  • 使用@autorelease块代替NSAutoreleasePool
  • 不能使用区域(NSZone)
  • 对象型变量不能作为C语言结构体的成员
  • 显式转换id和void*

不要显式调用dealloc

dealloc 方法在大多数情况下还适用于删除已注册的代理或观察者对象。
在这里插入图片描述

五、属性

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

webstorm 使用prettier格式化保存 导致代码缩进与gitlab代码不一致问题

问题 webstorm显示缩进正常 gitlab显示不正常 解决 .prettierrc.js module.exports {printWidth: 100,tabWidth: 2,useTabs: false, //设置为false 不使用tab作为缩进符semi: true,vueIndentScriptAndStyle: true,singleQuote: true,quoteProps: as-needed,bracketSpaci…

肖恩的投球游戏——前缀和

题目链接:1.肖恩的投球游戏 - 蓝桥云课 (lanqiao.cn) 前缀和: package lanqiao;import java.util.Arrays; import java.util.Scanner;/*** 2023/11/29* 前缀和问题*/ public class lanqiao3693_肖恩的投球游戏 {public static void main(String[] args) …

vue3+vite项目打包遇到的问题

一、项目打包出现空白页 vite.config.js中,添加base: ./ import { defineConfig } from vite import vue from vitejs/plugin-vueexport default defineConfig({base: ./, })router/index.js,将路由模式改成hash模式 import { createRouter, createWe…

性能分析调优模型

性能测试除了为获取性能指标外,更多是为了发现性能瓶颈和性能问题,然后针对性能问题和性能瓶颈进行分析和调优。在当今互联网高速发展的时代,结合传统软件系统模型以及互联网网站特征,性能调优的模型可以归纳总结为如图1-5-1所示的…

idea 的基本配置

一、安装目录介绍 其中:bin 目录下: 二、配置信息目录结构 这是 IDEA 的各种配置的保存目录。这个设置目录有一个特性,就是你删除掉整个目录之后,重新启动 IntelliJ IDEA 会再自动帮你生成一个全新的默认配置,所以很多…

力扣● 583. 两个字符串的删除操作 ● 72. 编辑距离 ● 编辑距离总结篇

● 583. 两个字符串的删除操作 注意审题: 给定两个单词 word1 和 word2 ,返回使得 word1 和 word2 相同所需的最小步数。 每步 可以删除任意一个字符串中的一个字符。 删除最少的字符使两者相同,说明留下来的就是最大公共子序列。不要求…

01.Vue2入门

一、为什么要学习Vue 1.前端必备技能 2.岗位多,绝大互联网公司都在使用Vue 3.提高开发效率 4.高薪必备技能(Vue2Vue3) 二、什么是Vue 概念:Vue (读音 /vjuː/,类似于 view) 是一套 **构建用户界面 ** 的 渐进式 …

多种双拼方案的实现

首发日期 2024-03-14, 以下为原文内容: 就像 GNU/Linux 用户, 虽然比例小, 却又分散为一堆不同的发行版. 双拼用户在拼音输入法之中的比例也很小, 同时也分为各种不同的双拼方案. 那么作为一个 双拼 输入法, 最重要的事情是什么呢 ? 嗯, 那当然是支持自定义双拼方案 ! 实际上…

网络协议与层次划分:探索计算机网络体系结构

✨✨ 欢迎大家来访Srlua的博文(づ ̄3 ̄)づ╭❤~✨✨ 🌟🌟 欢迎各位亲爱的读者,感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢,在这里我会分享我的知识和经验。&am…

AWTK slider_circle 控件发布

slider_circle 控件。 主要特色: 支持正向和反向支持设置滑块的半径支持背景线宽和颜色支持前景线宽和颜色支持设置是否显示值的文本支持设置起始角度和结束角度支持设置格式化值的格式字符串支持使用图片填充背景和前景 界面效果: 注意: …

【绘图案例-绘图的方式1 Objective-C语言】

一、接下来,我们来说这个,绘图的方式 1.新建一个项目,Name:04-绘图的方式, 方式:就是,我要同样画一条线,然后,用不同的代码,把它写出来,这就叫方式, 我们在storyboard里边,还拖一个UIView,这些步骤都一样, 我们来一个,宽= 300, 高 = 300 , 然后,再来一个水…

zabbix配置

1 下载zabbix 1 配置yum源 rpm -Uvh https://repo.zabbix.com/zabbix/5.0/rhel/7/x86_64/zabbix-release- 5.0-1.el7.noarch.rpm yum clean all yum makecache fast 完成后会出现zabbix.repo文件 2安装zabbix服务 yum -y install zabbix-server-mysql zabbix-web-mysql z…

计算机网络——物理层(信道复用技术)

计算机网络——物理层(信道复用技术) 信道复用技术频分多址与时分多址 频分复用 FDM (Frequency Division Multiplexing)时分复用 TDM (Time Division Multiplexing)统计时分复用 STDM (Statistic TDM)波分复用码分复用 我们今天接着来看信道复用技术&am…

20W-50W厚膜无感电阻TO-220封装技术规格散热说明

EAK为设计工程师提供了一种开放式屏蔽基板器件,用于需要卓越热性能的应用,开发了一种额定功率高达 50W 的厚膜功率电阻器。该电阻器采用 TO-220 开放式屏蔽基板封装,并具有与基板粘合的绝缘锥形文丘里管,以实现最大的散热。 电阻器…

鸿蒙Harmony应用开发—ArkTS声明式开发(容器组件:Counter)

计数器组件,提供相应的增加或者减少的计数操作。 说明: 该组件从API Version 7开始支持。后续版本如有新增内容,则采用上角标单独标记该内容的起始版本。 子组件 可以包含子组件。 接口 Counter() 从API version 9开始,该接口…

Flask 专题

[CISCN2019 总决赛 Day1 Web3]Flask Message Board 查看session解密 但不知道密钥,题目说FLASK,那肯定就是找密钥,发现输入什么都没有显示,只有author那里有回显在版上,所以尝试sstl,{{config}}找到密钥 扫目录发现有admin进入…

Python数学建模-2.5Pandas库介绍

2.5.1Pandas基本操作 Pandas是一个强大的Python数据分析库,它提供了快速、灵活且富有表现力的数据结构,设计初衷是为了处理关系型或标记型数据。Pandas的基本操作涵盖了数据的读取、处理、筛选、排序、分组、合并以及可视化等多个方面。 以下是一些Pan…

判断闰年(C语言)

一、运行结果&#xff1b; 二、源代码&#xff1b; # define _CRT_SECURE_NO_WARNINGS # include <stdio.h>int main() {//初始化变量值&#xff1b;int year 2000;//执行循环判断&#xff1b;while (year < 2010){//执行流程&#xff1b;//判断能否整除4&#xff1…

配置IPv4静态路由示例

配置IPv4静态路由示例 图1 配置IP静态路由组网图 组网需求配置思路操作步骤配置文件 组网需求 如图1所示&#xff0c;STA1、STA2和PC1属于不同网段&#xff0c;STA1在AP1中上线&#xff0c;STA2在AP2中上线&#xff0c;要求配置静态路由&#xff0c;使PC1与STA1和STA2能够互…

python之万花尺

1、使用模块 import sys, random, argparse import numpy as np import math import turtle import random from PIL import Image from datetime import datetime from math import gcd 依次使用pip下载即可 2、代码 import sys, random, argparse import numpy as np imp…