Swift中如何避免循环引用的方法

内存管理中经常会遇到的一个问题便是循环引用。首先,我们来了解一下iOS是如何进行内存管理的。

和OC一样,swift也是使用自动引用计数ARC(Auto Reference Counteting)来自动管理内存的,所以我们不需要过多考虑内存管理.当某个类实例不需要用到的时候,ARC会自动释放其占用的内存.

ARC

ARC(Automatic Reference Counting) 是苹果的自动内存管理机制。正如其名:自动引用计数,根据引用计数来决定内存块是否应该被释放。

当一个对象被创建的时候,它的引用计数为1。在它的生命周期内,其引用计数可以增加或减少。当它的引用计数减为0的时候,其所占用内存便会被释放。其生命周期如图所示:

强引用和弱引用(Strong/Weak References)

定义一个变量的时候可以声明其strong和weak属性,默认是strong类型。

struct Example {
 var strongView = UIView()
 weak var weakView = UIView()
}

强引用和弱引用有什么不同呢?

强引用会使变量的引用计数加1。如果一个对象的引用计数为2,当它再次被强引用的时候,它的引用计数会变为3。

弱引用不会增加引用计数。如果一个对象的引用计数为2,当它再次被弱引用的时候,它的引用计数仍为2。

强引用的对象能保证其被调用的时候仍在内存中,而弱引用不行。

循环引用和内存泄漏

当A引用B中的成员变量,而B又对A中的成员变量有引用的时候就会发生循环引用。
比如:

class Book {
 private var pages = [Page]()

 func add(_ page : Page) {
  pages.append(page)
 }
}

class Page {
 private var book : Book

 required init(book : Book) {
  self.book = book
 }
}

let book = Book()
let page = Page(book: book)
book.add(page)

此时,book对page有强引用,同时page对book也有强引用。这个时候便有循环引用,会导致内存泄漏。

对于这种两个变量的相互强引用导致的内存泄漏该如何解决呢?

Structs 和 Classes

正确的使用struct 和 class能避免循环引用的发生。

struct 和 class 都有成员变量,函数和协议。那么,它们之间有什么区别呢?

struct 是 值类型。
class 是 引用类型。

当引用或者传递 值类型 变量的时候,它会在内存中重新分配地址,copy内容到新的地址中。

struct Element {
 var name : String
 var number : Int
}

var firstElement = Element(name: "A", number: 1)

var secondElement = firstElement
secondElement.name = "B"
secondElement.number = 2

print(firstElement)
print(secondElement)

输出的结果为:

Element(name: “A”, number: 1)
Element(name: “B”, number: 2)

当引用或者传递 引用类型 变量的时候,新的变量指针指向的仍是原先的内存地址。此时原先的变量值改变的话,也会导致新变量值的变化。

比如:

class Element {
 var name : String
 var number : Int

 required init(name : String, number : Int) {
  self.name = name
  self.number = number
 }
}

extension Element : CustomStringConvertible {
 var description : String {
  return "Element(name: \(name), number: \(number))"
 }
}

var firstElement = Element(name: "A", number: 1)

var secondElement = firstElement
secondElement.name = "B"
secondElement.number = 2

print(firstElement)
print(secondElement)

此时的输出结果为:

Element(name: B, number: 2)
Element(name: B, number: 2)

我们为什么在此讨论值类型和引用类型呢?

回到之前book和pages的例子。我们用struct代替class:

struct Book {
 private var pages = [Page]()

 mutating func add(_ page : Page) {
  pages.append(page)
 }
}

struct Page {
 private var book : Book

 init(book : Book) {
  self.book = book
 }
}

var book = Book()
let page = Page(book: book)
book.add(page)

此时,便不会发生循环引用的情况。

如果仍想使用class的话,可以使用weak来避免循环引用:

class Book {
 private var pages = [Page]()

 func add(_ page : Page) {
  pages.append(page)
 }
}

class Page {
 private weak var book : Book?

 required init(book : Book) {
  self.book = book
 }
}

let book = Book()
let page = Page(book: book)
book.add(page)

Protocols

Protocols在swift中使用的很广泛。class,struct 和 enum 都可以使用Protocol。但是如果使用不当的话,同样会引起循环引用。

比如:

protocol ListViewControllerDelegate {
 func configure(with list : [Any])
}

class ListViewController : UIViewController {

 var delegate : ListViewControllerDelegate?

 override func viewDidLoad() {
  super.viewDidLoad()
 }

}

ListViewController 中的delegate变量是strong类型的,可以引用任何实现它protocol的变量。假如实现其protocol的变量对该 view controller 同样有强引用的话会怎么样? 声明delegate为weak可能会避免这种情况,但是这样的话会引起编译错误,因为structs和enums不能引用weak变量。

该如何解决呢?当声明protocol的时候,我们可以指定只有class类型的变量可以代理它,这样的话就可以使用weak来修饰了。

protocol ListViewControllerDelegate : class {
 func configure(with list : [Any])
}

class ListViewController : UIViewController {

 weak var delegate : ListViewControllerDelegate?

 override func viewDidLoad() {
  super.viewDidLoad()
 }

}

Closures

Closures 导致循环引用的原因是:Closures对使用它们的对象有一个强引用。

比如:

class Example {

 private var counter = 0

 private var closure : (() -> ()) = { }

 init() {
  closure = {
   self.counter += 1
   print(self.counter)
  }
 }

 func foo() {
  closure()
 }

}

此时,对象对closure有一个强引用,同时在closure的代码块中又对该对象本身有一个强引用。这样就引起了循环引用的发生。

这种情况,可以有两种方法来解决这个问题。

1.使用[unowned self]:

class Example {

 private var counter = 1

 private var closure : (() -> ()) = { }

 init() {
  closure = { [unowned self] in
   self.counter += 1
   print(self.counter)
  }
 }

 func foo() {
  closure()
 }

}

使用[unowned self] 的时候需要注意的一点是:调用closure的时候如果对象已经被释放的话,会出现crash。

2.使用[weak self]:

class Example {

 private var counter = 1

 private var closure : (() -> ()) = { }

 init() {
  closure = { [weak self] in
   self?.counter += 1
   print(self?.counter ?? "")
  }
 }

 func foo() {
  closure()
 }

}

[weak self] 和[unowned self] 的区别是 [weak self]处理的时候是一个可选类型。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Swift循环遍历集合的方法总结分享

    前言 之前分享总结过OC循环遍历,文章点击这里:iOS遍历集合(NSArray,NSDictionary.NSSet)方法总结.随着Swift的逐渐完善,自己使用Swift开发的项目经验和知识逐渐积累,是时候总结一下Swift的循环遍历了.相信Swift一定会给你一些不一样的东西,甚至是惊喜,感兴趣的朋友们下面来一起看看吧. 第一种方式:for-in循环 OC延续了C语言的for循环,在Swift中被彻底改造,我们无法再使用传统形式的for循环了 遍历数组和字典: //遍历数组 let iosA

  • Swift中的条件判断、循环、跳转语句基础学习笔记

    一.引言 一种编程语言的强大与否,很大程度上取决于其提供的程序流程控制方案,就如使用汇编语言实现复杂的程序流程是一件痛苦的事情.Swift中提供了许多强大的流程控制语句,例如快速遍历for-in,while循环,repeat-while循环,switch选择等,需要注意的是,在Swift2.2中,for(a;b;c)循环已经被弃用掉,并且Swift中的Switch语句也更加强大,可以处理任意数据类型. 二.for-in循环 配合范围运算符,for-in循环可以用来执行确定次数的循环,示例如下:

  • Swift学习笔记之逻辑分支与循环体

    分支的介绍 分支即if/switch/三目运算符等判断语句 通过分支语句可以控制程序的执行流程 1.if OC 后面条件必须加() 后面提条件非0即真 如果只有一条if后面的大括号可省略 if(a>0)NSlog(@"yes"); Swift if 后面不加括号 if 后面条件必须是明确的Bool类型 即使只有一条指令if后面的大括号亦不可省略 if else 的使用与OC一致,只是条件语句后不加括号;三目运算符和OC基本一致; 2.guard guard 是swift2.0 新

  • Swift中循环语句中的转移语句 break 和 continue

    下面通过实例代码给大家介绍了Swift中循环语句中的转移语句 break 和 continue,具体代码如下所示: /** 循环语句中的转移语句 break 和 continue */ let array:Array = [3, 4, 5, 6, 7, 8, 9] for k in array { if k == 5 { print(k) break } } print("--------->") for k in array { if k == 5 { // 结束本次循环,进入

  • 详解Swift语言的while循环结构

    Swift 编程语言中的 while 循环语句只要给定的条件为真时,重复执行一个目标语句. 语法 Swift 编程语言的 while 循环的语法是: 复制代码 代码如下: while condition {    statement(s) } 这里 statement(s) 可以是单个语句或语句块.condition 可以是任何表达式.循环迭代当条件(condition)是真的. 当条件为假,则程序控制进到紧接在循环之后的行. 数字0,字符串"0"和"",空列表 l

  • Swift流程控制之循环语句和判断语句详解

    Swift提供了所有c类语言的控制流结构.包括for和while循环来执行一个任务多次:if和switch语句来执行确定的条件下不同的分支的代码:break和continue关键字能将运行流程转到你代码的另一个点上. 除了C语言传统的for-condition-increment循环,Swift加入了for-in循环,能更加容易的遍历arrays, dictionaries, ranges, strings等其他序列类型. Swift的switch语句也比C语言的要强大很多. Swift中swi

  • 详解Swift编程中的for循环的编写方法

    for 循环是一个循环控制结构,可以有效地编写来执行的特定次数的循环. 语法 for 循环在 Swift 编程语言的语法是: 复制代码 代码如下: for init; condition; increment{    statement(s) } 下面是在一个循环的流程控制: 初始化 init 步骤首先被执行,并且仅一次.在这一步,可以声明和初始化任何循环控制变量. 只要一个分号出现,不需要一定把一个语句放在这里. 接下来,计算条件.如果为真,则执行循环体.如果是假,循环体不执行,只是在 for

  • swift中c风格的for循环执行效率

    今天用swift写了1至99 9999的和,测试其执行效率,但是发现不同代码执行效率大大不同 1. 2. 从结果可以看到,执行速度相差5倍多,若数据再大点,就会很 明显了.这说明不同代码风格执行效率不同,明显使用c风格的for循环执行速度会更快些. 而对于即将出现的正式版swift 3.0废除c风格for循环一事,你怎么看待? 以上所述是小编给大家介绍的swift中c风格的for循环执行效率 ,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的.在此也非常感谢大家对我们网站的

  • Swift中如何避免循环引用的方法

    内存管理中经常会遇到的一个问题便是循环引用.首先,我们来了解一下iOS是如何进行内存管理的. 和OC一样,swift也是使用自动引用计数ARC(Auto Reference Counteting)来自动管理内存的,所以我们不需要过多考虑内存管理.当某个类实例不需要用到的时候,ARC会自动释放其占用的内存. ARC ARC(Automatic Reference Counting) 是苹果的自动内存管理机制.正如其名:自动引用计数,根据引用计数来决定内存块是否应该被释放. 当一个对象被创建的时候,

  • IOS 避免self循环引用的方法的实例详解

    IOS 避免self循环引用的方法的实例详解 示例代码: // - weak & strong #define myWeakify(VAR) \ try {} @finally {} \ __weak __typeof__(VAR) VAR##_myWeak_ = (VAR) #define myStrongify(VAR) \ try {} @finally {} \ __strong __typeof__(VAR) VAR = VAR##_myWeak_ #define myStrongif

  • MySQL存储过程中使用WHILE循环语句的方法

    本文实例讲述了MySQL存储过程中使用WHILE循环语句的方法.分享给大家供大家参考.具体如下: mysql> mysql> delimiter $$ mysql> mysql> CREATE PROCEDURE myProc() -> BEGIN -> -> DECLARE i int; -> SET i=1; -> loop1: WHILE i<=10 DO -> IF MOD(i,2)<>0 THEN /*Even num

  • jquery中object对象循环遍历的方法

    一个朋友问对象如何转为数组,当我问他为啥要转得时候,他告诉我,数组可以用js循环遍历,而对象则不可以.其实呢,对象同样可以循环遍历的啊.不用转换也可以循环!说明你对js或者jquery的某些操作不是很熟练!在这里我简单介绍一下! 案例 我们看如下对象: var data={ 张三:69, 李四:72, 王五:90, 二麻子:88, 前端博客:100, haorooms : 98, 王大壮:99 } 假如上面是后台返回的一个key,一个是value的对象(这种对象相信大家经常遇到吧!),现在要把这

  • 详解在springmvc中解决FastJson循环引用的问题

    我们先来看一个例子: package com.elong.bms; import java.io.OutputStream; import java.util.HashMap; import java.util.Map; import com.alibaba.fastjson.JSON; public class Test { public static void main(String[] args) { Map<String, Student> maps = new HashMap<

  • swift中可选值?和!使用的方法示例

    Optional 可选值 Optional是 Swift 的一大特色,也是 Swift 初学者最容易困惑的问题. 定义变量时,如果指定该变量是可选的,表示该变量可以有一个指定类型的值,也可以是 nil. 此外,Swift的nil也和Objective-C有些不一样,在Objective-C中,只有对象才能为nil,而在Swift里,当基础类型(整形.浮点.布尔等)没有值时,也是nil,而不是一个初始值,没有初始值的值,是不能使用的,这就产生了Optional类型.定义一个Optional的值很容

  • 在 Swift 中编写Git Hooks脚本的方法

    目录 前言 用git hooks自动生成提交信息 为什么我使用Swift? 让我们开始吧 编写git钩子 检索提交消息 注意: 检索问题编号 修改提交信息 设置git钩子 测试结果 参考资料 前言 这周,我决定完成因为工作而推迟了一周的TODO事项来改进我的Git工作流程. 为了在提交的时候尽可能多的携带上下文信息,我们让提交信息包含了正在处理的JIRA编号.这样,将来如果有人回到我们现在正在提交的源代码,输入​ ​git blame​ ​,就能很容易的找出JIRA的编号. 每次提交都包含这些信

  • Swift中字典与JSON转换的方法

    Swift中经常会遇到字典和字符串的相互转换,因此可以转换可以封装起来,转换代码如下: func convertStringToDictionary(text: String) -> [String:AnyObject]? { if let data = text.data(using: String.Encoding.utf8) { do { return try JSONSerialization.jsonObject(with: data, options: [JSONSerializat

  • Swift中图片资源使用流程的优化方法详解

    前言 去年发布的Xcode9支持在代码编辑中直接插入图片,类似如下效果 但用了一段时间以后还是不太喜欢,换回了原来的方法. 本篇中许多实现细节已经在 iOS中多语言本地化流程的优化中写过,没再重复,若有疑问建议先阅读该文章. 传统的方法 // iOS let closeImage = UIImage(named: "close") // macOS let closeImage = NSImage(named: NSImage.Name("close")) 是不是看

随机推荐