Swift是自动管理内存的。这意味着,你不需要主动释放内存。
比如Foo内包含的Bar,可以随同Foo一起被释放:
import UIKit
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {var window : UIWindow?func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {Foo()return true}
}
class Foo {let bar: Barinit() {bar = Bar()}deinit {print("Foo exit")}
}
class Bar {deinit {print("Bar exit")}
}复制代码
执行此代码,会打印:
Foo exit
Bar exit复制代码
可见Foo和Bar都是自动释放的。作为程序员,你不需要做任何内存的主动释放。
但是,有一种特殊情况,叫做双向引用,导致释放A时,需要释放B,而B又引用了A,那么两个都无法被释放:
class Foo {let bar: Barinit() {bar = Bar()bar.foo = self}deinit {print("Foo exit")}
}
class Bar {var foo: Foo? = nildeinit {print("Bar exit")}
}复制代码
此代码只会打印:
App exit复制代码
此时,需要做的就是把这个双向引用中的一个设置为weak,表示的意思是尽管我持有这个引用,但是释放的时候,却无需考虑此对象的释放。
import UIKit
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {var window : UIWindow?func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {Baz()print("App exit")return true}
}
typealias Bar = (()->Void)
class Foo {func work(_ bar : Bar) {bar()}deinit {print("Foo exit")}
}
class Baz {var a : String?init (){a = "1"let f = Foo()f.work(){[weak self]() inprint(self?.a)}}
}复制代码
当然,不标记也是不行的,因为编译器就不会通过,它要求只要引用了self,就必须标记。