Swift内存管理&闭包使用

2019-12-11  本文已影响0人  woniu

一、内存管理

1、跟OC一样,Swift也是采取基于引用计数的ARC内存管理方案(针对堆空间)
2、Swift的ARC中有三种引用
   class PersonX { }
   weak var person0 : PersonX?
   weak var person1 : AnyObject?
        
   unowned var person3 : PersonX?
   unowned var person4 : AnyObject?

二、Autoreleasepool

我们可以看出,Swift和OC的autoreleasepool使用没什么不同。

 for _ in 0...5000 {
     autoreleasepool {                
     }
 }

三、循环引用

1、weak、unowned都能解决循环引用的问题,unowned要比weak少些性能消耗。

a.生命周期中可能会变为nil的使用weak。
b.初始化赋值后再也不会变为nil的使用unowned。
c.这里有疑问,等到后面补充一下:

四、闭包循环引用

1、 闭包表达式默认会对用到的外层对象产生额外的强应用。(对外层对象进行了retain操作)
class PersonY {
    var fn: (() -> ())?
    func run() {
        print("run")
    }
    deinit { print("deinit") }
}

错误调用:

  func test() {
            let p = PersonY()
            p.fn = {
                p.run()
            }
        }
  test()

修正方法:

  func test() {
            let p = PersonY()
            p.fn = { [weak p] in
                    //或者[unowned p] in
                p!.run()
            }
        }
   test()
2、如果想在定义闭包属性的同时引用self,这个闭包必须是lazy的(因为在实例方法初始化完毕之后才能引用self)
class PersonZ {
    lazy var fn:(() -> ()) = {
        [weak self] in
        self?.run()
    }
    func run() {
        print("run")
    }
    deinit {
        print("deinit")
    }
}
class PersonM {
    var age : Int = 0
    lazy var getAge : Int = {
        age
    }()
    deinit {
           print("deinit")
    }
}

五、@escaping

1、非逃逸闭包、逃逸闭包,一般都是当做参数传递给函数。

a.首先声明一个闭包类型:

typealias Fn = () -> ()

b.fn是非逃逸闭包

func test1(_ fn: Fn) {
    fn()
}

c.fn是逃逸闭包

 // fn是逃逸闭包
var gFn: Fn?
func test2(_ fn: @escaping Fn) {
    gFn = fn
}

 // fn是逃逸闭包
func test3(_ fn: @escaping Fn) {
    DispatchQueue.global().async {
        fn()
    }
}

class Person {
    var fn: Fn
    // fn是逃逸闭包
    init(fn: @escaping Fn) {
        self.fn = fn
    }
    func run() {
    // DispatchQueue.global().async也是一个逃逸闭包
    // 它用到了实例成员(属性、方法),编译器会强制要求明确写出self DispatchQueue.global().async {
    self.fn()
    }
}

d.逃逸闭包注意点:
逃逸闭包不可以捕获inout参数

typealias Fn = () -> ()
func other1(_ fn: Fn) {
    fn()
}
func other2(_ fn: @escaping Fn) {
    fn()
}
func test(value: inout Int) -> Fn {
    other1 { value += 1 }
    // error: 逃逸闭包不能捕获inout参数
        other2 { value += 1 }
    func plus() { value += 1 }
    // error: 逃逸闭包不能捕获inout参数 return plus
}
上一篇 下一篇

猜你喜欢

热点阅读