当前位置: 首页 > news >正文

Swift——自动引用计数ARC

ARC

ARC是swift使用的一种管理应用程序内存的机制,对于C语言我们知道,当我们申请一块空间,通常需要手动释放,不然会造成空间浪费,而有了ARC机制,你无需考虑内存的管理,因为ARC会在类的实例不再被使用时,自动释放内存空间。

ARC通常适用引用类型,比如类。

自动引用计数的规则:

  • 每创建一个类的实例对象,ARC就分配一块内存存储实例信息,引用计数+1
  • 当实例不再被使用,ARC自动释放实例所占内存,引用计数-1
  • 当引用计数为0时,实例被销毁。

eafe02e25f7c4256ba0dc37c46ebda3b.png

类实例之间的循环强引用

循环强引用:两个类实例都持有一个强引用的指向对方的属性

解决循环强引用方法:类之间的关系使用弱引用代替强引用。

303612e7d14e47098148a9f3c6655503.png

循环强引用示例:

class A{let aStr:Stringvar b:B?init(a: String) {self.aStr = a}deinit{print("A's deinit")}
}class B{var bStr:Stringvar a:A?init(str:String){self.bStr = str}deinit {print("B's deinit")}
}var objA:A?
var objB:B?objA = A(a: "AAAA")
objB = B(str: "BBBB")objA!.b = objB
objB!.a = objAobjA = nil
objB = nil
//由于objA.b还指向B,objcB.a还指向A所以两者的实例还未被释放,此时打印无结果

此时如果要释放A和B只能这么做:

5be889ae49d9439ba870cee566853113.png

一般解决该办法之一是通过弱引用weak,弱引用不会增加ARC计数。

因此可以改成:

class A{let aStr:Stringweak var b:B?//使用弱引用init(a: String) {self.aStr = a}deinit{print("A's deinit")}
}class B{var bStr:Stringweak var a:A?//使用弱引用init(str:String){self.bStr = str}deinit {print("B's deinit")}
}var objA:A?
var objB:B?objA = A(a: "AAAA")
objB = B(str: "BBBB")objA!.b = objB
objB!.a = objAobjA = nil//此时A释放
objB = nil//此时B释放

无主引用Unowned

解决循环引用的另一种方式就是无主引用,无主引用修饰的实例属性与引用它的实例有着相同的生命周期

  • 在声明属性或者变量时,在前面加上关键字unowned表示这是一个无主引用
  • 使用无主引用,必须确保引用始终指向一个未销毁的实例,这也意味着无主引用的对象有确定的值。
  • 如果试图在实例被销毁后,访问该实例的无主引用,会触发运行时错

0efe629cb31c4502aa0d294e729034fb.png

class A{let aStr:Stringvar b:B?//使用弱引用init(a: String) {self.aStr = a}deinit{print("A's deinit")}
}class B{var bStr:Stringunowned var a:A?//使用弱引用init(bStr: String, a: A? = nil) {self.bStr = bStrself.a = a}deinit {print("B's deinit")}
}var objA:A?objA = A(a: "AAAA")objA!.b = B(bStr: "bbbb",a:objA)objA = nil
//这里会释放A和B,因为B里的a是无主引用,类似于弱引用,这样就没有指向A的对象了,A被释放,A里面的b也被销毁,指向B的对象也没有了,B被释放

闭包引起的循环强引用

将一个闭包赋值给类实例的某个属性,并且这个闭包体中又使用了这个类实例时。这个闭包体中可能访问了实例的某个属性,或者闭包中调用了实例的某个方法,这两种情况都导致了闭包“捕获”self,从而产生了循环强引用。

b80ac61229e34ca99609308367f66c1b.png

例如:


class A{let aStr:Stringlet isShow:Boollazy var closures:()->String = {if self.isShow {return self.aStr}else{return "isShow is False"}}init(aStr: String, isShow: Bool) {self.aStr = aStrself.isShow = isShow}deinit{print("A's deinit")}}var objA:A?
objA = A(aStr: "AAAA", isShow: true)var value:String = objA!.closures()
print(value)objA = nil

解决办法跟类实例循环引用方法一样,声明每一个捕获引用为弱引用或者无主引用。

  • 弱引用:在被捕获的引用可能会变为nil时,将闭包内的捕获定义为弱引用
  • 无主引用 :在闭包和捕获的实例总是互相引用并且总是同时销毁时,将闭包内的捕获定义为无主引用
  • 如果被捕获的引用绝对不会变为nil,应该用无主引用,而不是弱引用

5912f98066dd43cba93cc22dbaf07588.png

示例:

class A{let aStr:Stringlet isShow:Boollazy var closures:()->String = {//捕获列表是[unowned self],表示将self捕获为无主引用而不是强引用[unowned self] inif self!.isShow {return self!.aStr}else{return "isShow is False"}}init(aStr: String, isShow: Bool) {self.aStr = aStrself.isShow = isShow}deinit{print("A's deinit")}}var objA:A?
objA = A(aStr: "AAAA", isShow: true)var value:String = objA!.closures()
print(value)objA = nil
//这里会释放A

 

 

 

 


http://www.mrgr.cn/news/78308.html

相关文章:

  • Android 单选框RadioGroup+RadioButton
  • GM、BP、LSTM时间预测预测代码
  • Spring框架整合单元测试
  • H.265流媒体播放器EasyPlayer.js无插件H5播放器关于移动端(H5)切换网络的时候,播放器会触发什么事件
  • vxe-modal VxeUI 窗口组件弹窗多窗口模式
  • Redis 可观测最佳实践
  • Javascript Insights: Visualizing Var, Let, And Const In 2024
  • Hbase2.2.7集群部署
  • 【不定长滑动窗口】【灵神题单】【刷题笔记】
  • 【拥抱AI】RAG如何通过分析反馈、识别问题来提高命中率
  • 探索.NET世界的无限可能——带你轻松了解.NET
  • Scala—Map用法详解
  • 图元交互设计
  • 【去毛刺】OpenCV图像处理基础:腐蚀与膨胀操作入门
  • 365天深度学习训练营-第P6周:VGG-16算法-Pytorch实现人脸识别
  • digit_eye开发记录(2): Python读取MNIST数据集
  • 大语言模型LLM的微调中 QA 转换的小工具 txt2excel.py
  • Java AQS(AbstractQueuedSynchronizer):深入剖析
  • v-for产生 You may have an infinite update loop in a component render function
  • 直言抖音电商环境恶化,叶国富也想指点张一鸣
  • 【拥抱AI】RAG如何提高向量化的质量
  • 关于node全栈项目打包发布linux项目问题总集
  • SQL基础入门—— 简单查询与条件筛选
  • ubuntu 安装docker
  • Linux下的火墙管理及优化
  • C语言蓝桥杯组题目