Swift語法 Swift5 【12 - 初始化器init】


  • 作者: Liwx
  • 郵箱: 1032282633@qq.com
  • 源碼: 需要源碼的同學(xué), 可以在評論區(qū)留下您的郵箱

iOS Swift 語法 底層原理內(nèi)存管理分析 專題:【iOS Swift5語法】

00 - 匯編
01 - 基礎(chǔ)語法
02 - 流程控制
03 - 函數(shù)
04 - 枚舉
05 - 可選項(xiàng)
06 - 結(jié)構(gòu)體和類
07 - 閉包
08 - 屬性
09 - 方法
10 - 下標(biāo)
11 - 繼承
12 - 初始化器init
13 - 可選項(xiàng)


目錄

  • 01-初始化器
  • 02-初始化器的相互調(diào)用
  • 03-兩段式初始化
  • 04-安全檢查
  • 05-重寫
  • 06-自動(dòng)繼承
  • 07-required
  • 08-屬性觀察器
  • 09-可失敗初始化器
  • 10-反初始化器(deinit)

01-初始化器

  • 類肪笋、結(jié)構(gòu)體墙懂、枚舉都可以定義初始化器
  • 2種初始化器: 指定初始化器(designated initializer)、便捷初始化器(convenience initializer)
    • 如果有自定義初始化器, 編譯器 不會(huì)為類自動(dòng)生成無參初始化器(默認(rèn)初始化器)
    • 每個(gè)類至少有一個(gè)指定初始化器,指定初始化器是類的主要初始化器
    • 默認(rèn)初始化器總是類的指定初始化器
    • 類偏向于少量指定初始化器, 一個(gè)類通常只有一個(gè)指定初始化器

  • 初始化器的相互調(diào)用規(guī)則
    • 指定初始化器必須從它的直系父類調(diào)用指定初始化器
    • 便捷初始化器必須從相同的類調(diào)用另一個(gè)初始化器
    • 便捷初始化器最終必須調(diào)用一個(gè)指定初始化器

  • 便捷初始化器聲明: 用convenience關(guān)鍵字修飾便捷初始化器
// 指定初始化器
init(parameters) {
    statements
}

// 便捷初始化器
convenience init(parameters) {
    statements
}

  • 如果沒有自定義初始化器, 編譯器會(huì)為類自動(dòng)生成無參初始化器(默認(rèn)初始化器)
class Size {
    var width: Int = 0
    var height: Int = 0
//    init() {    // 編譯器生成這種無參初始化器
//
//    }
}
var size = Size()

  • 如果有自定義初始化器, 編譯器 不會(huì)為類自動(dòng)生成無參初始化器(默認(rèn)初始化器)
class Size {
    var width: Int = 0
    var height: Int = 0
    init(width: Int, height: Int) {    // 編譯器生成這種無參初始化器
        self.width = width
        self.height = height
    }
}
//var size = Size()   // error: missing arguments for parameters 'width', 'height' in call
var size = Size(width: 10, height: 20)

  • 便捷初始化器聲明: 用convenience關(guān)鍵字修飾便捷初始化器
    • 便捷初始化器必須從相同的類調(diào)用另一個(gè)初始化器
    • 便捷初始化器最終必須調(diào)用一個(gè)指定初始化器
class Size {
    var width: Int = 0
    var height: Int = 0
    
    init() {
        
    }
    
    convenience init(width: Int, height: Int) {
        // 便捷最終必須調(diào)用一個(gè)指定初始化器
        self.init()         // 如果沒調(diào)用self.init(), error: 'self' used before 'self.init' call or assignment to 'self'
        self.width = width
        self.height = height
    }
}
var s1 = Size()
var s2 = Size(width: 10, height: 10)

  • 指定初始化器
class Size {
    var width: Int = 0
    var height: Int = 0
    
    // 指定初始化器(主要初始化器)
    init(width: Int, height: Int) {
        self.width = width
        self.height = height
    }
    
    convenience init(width: Int) {
        self.init(width: width, height: 0) // 如果沒調(diào)用指定初始化器 error: 'self' used before 'self.init' call or assignment to 'self'
        self.width = width
    }
    
    convenience init(height: Int) {
        self.init(width: 0, height: height)
        self.height = height
    }
    
    convenience init() {
        self.init(width: 0, height: 0)
    }
}
var s1 = Size()
var s2 = Size(width: 10, height: 10)
var s3 = Size(width: 10)
var s4 = Size(height: 20)

  • 指定初始化器必須從它的直系父類調(diào)用指定初始化器
    • 子類的指定初始化器只能調(diào)用直系父類的初始化器
    • 子類指定初始化器不能直接調(diào)用本身`的指定初始化器
class Person {
    var age: Int = 0
    init(age: Int) {
        self.age = age
    }
    
    convenience init() {
        self.init(age: 0)
    }
}

class Student : Person {    // 類繼承冒號左右最好空一格
    var score: Int
    init(age: Int, score: Int) {
        
        // 指定初始化器必須從它的直系父類調(diào)用指定初始化器
        self.score = score // 注意: 不能寫到super.init(age: age)之后, 否則報(bào)錯(cuò) error: property 'self.score' not initialized at super.init call
        super.init(age: age)
        
        // error: 'super.init' isn't called on all paths before returning from initializer
//        self.age = age
//        self.score = score
    }
    
    
    init() {
        
        // 子類的指定初始化器不能直接調(diào)用本身的指定初始化器
//        self.init(age: 0, score: 0) // error: designated initializer for 'Student' cannot delegate (with 'self.init'); did you mean this to be a convenience initializer?
        
        self.score = 0
        super.init(age: 0)
    }
}

//var stu1 = Student(age: 10) // 如果子類自定義初始化器,不會(huì)自動(dòng)繼承直系父類的初始化器,因?yàn)檫@樣會(huì)導(dǎo)致父類的存儲(chǔ)屬性為能初始化 error: missing argument for parameter 'score' in call
var stu2 = Student(age: 10, score: 20)

02-初始化器的相互調(diào)用

  • 初始化器的相互調(diào)用規(guī)則
    • 指定初始化器必須從它的直系父類調(diào)用指定初始化器
    • 便捷初始化器必須從相同的類調(diào)用另一個(gè)初始化器
  • 便捷初始化器最終必須調(diào)用一個(gè)指定初始化器

  • 這套規(guī)則保證了 使用任意初始化器,都可以完整地初始化實(shí)例
    • 總結(jié): 指定構(gòu)造器縱向調(diào)用,便捷初始化器橫向調(diào)用

QQ20200510-191425@2x.png

03-兩段式初始化

  • Swift在編碼安全方面是煞費(fèi)苦心,為了保證初始化過程的安全,設(shè)定了兩段式初始化媒殉、安全檢查
  • 兩段式初始化
    • 第1階段:初始化所有存儲(chǔ)屬性(向上調(diào)用)
      1.外層調(diào)用指定\便捷初始化器
      2.分配內(nèi)存給實(shí)例,但未初始化
      3.指定初始化器確保當(dāng)前類定義的存儲(chǔ)屬性都初始化
      4.指定初始化器調(diào)用父類的指定初始化器迁客,不斷向上調(diào)用,形成初始化器鏈
    • 第2階段:設(shè)置新的存儲(chǔ)屬性值(向下調(diào)用)
      1.從頂部初始化器往下,鏈中的每一個(gè)指定初始化器都有機(jī)會(huì)進(jìn)一步定制實(shí)例
      2.初始化器現(xiàn)在能夠使用self (訪問哀峻、修改它的屬性,調(diào)用它的實(shí)例方法等等)
      3.最終,鏈中任何便捷初始化器都有機(jī)會(huì)定制實(shí)例以及使用self
class Person {
    var age: Int = 0
    init(age: Int) {
        self.age = age
        // 此次可以個(gè)性化定制
    }
    
    convenience init() {
        self.init(age: 0)
        // 此次可以個(gè)性化定制
    }
}

class Student : Person {
    var score: Int
    init(age: Int, score: Int) {
        
        // 存儲(chǔ)屬性未完成初始化指針,不能用self
//        print(self) // error: 'self' used before 'super.init' call
//        self.age = 10   // error: 'self' used in property access 'age' before 'super.init' call
//        self.test() // error: 'self' used in method call 'test' before 'super.init' call
//        var fn = {  // error'self' captured by a closure before all members were initialized
//            self.age = 20
//        }
//        fn()
        
        self.score = score
        super.init(age: age)
        // 此次可以個(gè)性化定制
    }
    
    init() {
        self.score = 0
        super.init(age: 0)
        // 此次可以個(gè)性化定制
        self.test()
    }
    
    func test() {
        print("age: \(age), score: \(score)")
    }
}

04-安全檢查

  • 指定初始化器必須保證在調(diào)用父類初始化器之前,其所在類定義的所有存儲(chǔ)屬性都要初始化完成
  • 指定初始化器必須先調(diào)用父類指定初始化器,然后才能為繼承的屬性設(shè)置新值
  • 便捷初始化器必須先調(diào)用同類中的其它初始化器,然后再為任意屬性設(shè)置新值
  • 初始化器在第1階段初始化完成之前,不能調(diào)用任何實(shí)例方法不能讀取任何實(shí)例屬性的值,也不能引用self
  • 直到第1階段結(jié)束,實(shí)例才算完全合法

QQ20200510-201532@2x.png

class Person {
    var age: Int = 0
    init(age: Int) {
        self.age = age
        // 此次可以個(gè)性化定制
    }
    
    convenience init() {
        self.init(age: 0)
        // 此次可以個(gè)性化定制
    }
}

class Student : Person {
    var score: Int
    init(age: Int, score: Int) {
        self.score = score
        super.init(age: age)
    }
    
    init() {
        self.score = 0
//        self.age = 20   // 初始化器在`第1階段初始化完成之前`,`不能調(diào)用任何實(shí)例方法`哲泊、`不能讀取任何實(shí)例屬性的值`,也`不能引用self`
        super.init(age: 0)
        // `指定初始化器`必須`先調(diào)用父類指定初始化器`,然后才能為`繼承的屬性設(shè)置新值`
        self.age = 10   // age是從父類繼承的屬性
        self.test()
    }
    
    convenience init(score: Int) {
        self.init(age: 0, score: score)
        // `便捷初始化器`必須先調(diào)用`同類中的其它初始化器`,然后再為`任意屬性設(shè)置新值`
        self.age = 10
        self.score = 20
    }
    
    func test() {
        print("test")
    }
}

05-重寫

  • 當(dāng)重寫父類的指定初始化器時(shí),必須加上override (即使子類的實(shí)現(xiàn)是便捷初始化器)
    • 如果子類寫了一個(gè)匹配父類便捷初始化器的初始化器,不用加上override
    • 因?yàn)?code>父類的便捷初始化器永遠(yuǎn)不會(huì)通過子類直接調(diào)用,因此,嚴(yán)格來說,子類無法重寫父類的便捷初始化器
class Person {
    var age: Int = 0
    init(age: Int) {
        self.age = age
    }
    
    // 因?yàn)楦割惖谋憬莩跏蓟饔肋h(yuǎn)不會(huì)通過子類直接調(diào)用,因此,嚴(yán)格來說,子類無法重寫父類的便捷初始化器
    convenience init() {
        self.init(age: 0)
    }
}

class Student : Person {
    var score: Int
    
    init(age: Int, score: Int) {
        self.score = score
        super.init(age: age)
    }
    
    // 當(dāng)`重寫父類的指定初始化器`時(shí),必須加上`override` (`即使子類的實(shí)現(xiàn)是便捷初始化器`)
    // 重寫為指定初始化器
//    override init(age: Int) {
//        self.score = 0
//        super.init(age: age)
//    }
    
    // 重寫為便捷初始化器
//    override convenience init(age: Int) {
//        self.init(age: age, score: 0)
//
//    }
    
    // 如果`子類`寫了一個(gè)`匹配父類便捷初始化器`的初始化器,`不用加上override`
    // 子類寫了匹配父類的便捷初始化器的指定初始化器(不算重寫, 重寫表示調(diào)用super.init())
//    init() {
//        self.score = 0
//        super.init(age: 0)
//    }
    
    // 子類寫了匹配父類的便捷初始化器的便捷初始化器(不算重寫, 重寫表示調(diào)用super.init())
    convenience init() {
        self.init(age: 0, score: 0)
    }
    
}

06-自動(dòng)繼承

  • 1.如果子類沒有自定義任何指定初始化器,它會(huì)自動(dòng)繼承 父類所有的指定初始化器
  • 2.如果子類提供了父類所有指定初始化器的實(shí)現(xiàn)(要么通過方式1繼承,要么重寫)
    • 子類自動(dòng)繼承所有父類便捷初始化器
  • 3.就算子類添加了更多的便捷初始化器,這些規(guī)則仍然適用
  • 4.子類便捷初始化器的形式重寫父類的指定初始化器,也可以作為滿足規(guī)則2的一部分

  • 1.如果子類沒有自定義任何指定初始化器,它會(huì)自動(dòng)繼承父類所有的指定初始化器
class Person {
    var age: Int = 0
    var name: String
    
    init(age: Int, name: String) {
        self.age = age
        self.name = name
    }
    
    init(age: Int) {
        self.age = age
        self.name = ""
    }
    
    init() {
        self.age = 0
        self.name = ""
    }
}

class Student : Person {
    
}

var stu1 = Student(age: 10)
var stu2 = Student(age: 10, name: "Liwx")

  • 如果子類重寫了父類的指定初始化器,不會(huì)在自動(dòng)繼承父類的初始化器
class Person {
    var age: Int = 0
    var name: String
    
    init(age: Int, name: String) {
        self.age = age
        self.name = name
    }
    
    init(age: Int) {
        self.age = age
        self.name = ""
    }
    
    init() {
        self.age = 0
        self.name = ""
    }
}

class Student : Person {
    override init() {
        super.init()
    }
}
var stu1 = Student()
//var stu2 = Student(age: 10) // 如果子類重寫了初始化器,不會(huì)在自動(dòng)繼承父類的初始化器 error: argument passed to call that takes no arguments

  • 子類新增自定義便捷初始化器,如果子類沒有重寫父類初始化器,子類可以調(diào)用自動(dòng)繼承父類的初始化器
class Person {
    var age: Int = 0
    var name: String
    
    init(age: Int, name: String) {
        self.age = age
        self.name = name
    }
    
    init(age: Int) {
        self.age = age
        self.name = ""
    }
    
    init() {
        self.age = 0
        self.name = ""
    }
}

class Student : Person {
    
    var no: Int = 0 // 參數(shù)必須初始化值,才能保證父類繼承下來的初始化器能保證所有存儲(chǔ)屬性都設(shè)置了初始值
    convenience init(no: Int) {
        
//        self.no = no    // no存儲(chǔ)屬性未完成初始化, 不能調(diào)用self error: 'self' used before 'self.init' call or assignment to 'self'
        self.init(age: 0, name: "")
        // 所有存儲(chǔ)已經(jīng)初始化完成,可以個(gè)性化定制參數(shù)
        self.no = no
    }
}

var stu1 = Student()
var stu2 = Student(age: 10)
var stu3 = Student(age: 10, name: "Liwx")
var stu4 = Student(no: 20)

  • 2.如果子類提供了父類所有指定初始化器的實(shí)現(xiàn)(要么通過方式1繼承催蝗,要么重寫)
    • 子類自動(dòng)繼承所有父類便捷初始化器
class Person {
    var age: Int = 0
    var name: String
    
    init(age: Int, name: String) {
        self.age = age
        self.name = name
    }
    
    init() {
        self.age = 0
        self.name = ""
    }
    
    convenience init(age: Int) {
        self.init(age: age, name: "")
    }
    
    convenience init(name: String) {
        self.init(age: 0, name: name)
    }
}

class Student : Person {
    
    // 2.如果子類提供了父類所有指定初始化器的實(shí)現(xiàn)(要么通過方式1繼承切威,要么重寫)
    // `子類自動(dòng)繼承所有`的`父類便捷初始化器`
    
    // 子類重寫父類所有指定初始化器
    override init(age: Int, name: String) {
        super.init(age: age, name: name)
    }
    
    override init() {
        super.init()
    }
}

var stu1 = Student(age: 10, name: "Liwx")   // 子類提供了父類所有指定初始化器的實(shí)現(xiàn),`子類自動(dòng)繼承所有`的`父類便捷初始化器`
var stu2 = Student()
var stu3 = Student(age: 10)
var stu4 = Student(name: "Liwx")

  • 3.就算子類添加了更多的便捷初始化器,這些規(guī)則仍然適用
class Person {
    var age: Int = 0
    var name: String
    
    init(age: Int, name: String) {
        self.age = age
        self.name = name
    }
    
    init() {
        self.age = 0
        self.name = ""
    }
    
    convenience init(age: Int) {
        self.init(age: age, name: "")
    }
    
    convenience init(name: String) {
        self.init(age: 0, name: name)
    }
}

class Student : Person {
    convenience init(age: Int, name: String, no: Int) {
        self.init(age: age, name: name)
    }
}

  • 4.子類以便捷初始化器的形式重寫 父類的所有指定初始化器, 會(huì)自動(dòng)繼承父類的所有便捷初始化器
class Person {
    var age: Int = 0
    var name: String
    
    init(age: Int, name: String) {
        self.age = age
        self.name = name
    }
    
    init() {
        self.age = 0
        self.name = ""
    }
    
    convenience init(age: Int) {
        self.init(age: age, name: "")
    }
    
    convenience init(name: String) {
        self.init(age: 0, name: name)
    }
}

class Student : Person {
    
    init(age: Int, name: String, no: Int) {
        super.init(age: age, name: name)
    }
    
    // 子類以便捷初始化器的形式重寫父類的所有指定初始化器, 會(huì)自動(dòng)繼承父類的所有便捷初始化器
    convenience override init(age: Int, name: String) {
        self.init(age: age, name: name)
    }
    
    convenience override init() {
        self.init(age: 0, name: "")
    }
}

var stu1 = Student()
var stu2 = Student(age: 10)
var stu3 = Student(age: 10, name: "Liwx")
var stu4 = Student(age: 10, name: "Liwx", no: 20)
var stu5 = Student(name: "Liwx")

07-required

  • required修飾初始化器,表明其所有子類都必須實(shí)現(xiàn)該初始化器(通過繼承或者重寫實(shí)現(xiàn))
  • 如果子類重寫了required初始化器,也必須加上required ,不用加override

  • 通過自動(dòng)繼承方式, 會(huì)自動(dòng)繼承required修飾的指定初始化器
class Person {
    required init() { }
    init(age: Int) { }
}

class Student : Person {
    
}

var stu1 = Student()
var stu2 = Student(age: 10)

  • 子類新增指定初始化器,必須實(shí)現(xiàn)required修飾的指定初始化器
class Person {
    required init() { }
    init(age: Int) { }
}

class Student : Person {
    init(no: Int) {
        super.init()
    }
    required init() {
        super.init()
    }
}

var stu1 = Student()
var stu2 = Student(no: 20)
//var stu3 = Student(age: 10) // 子類重寫了父類指定初始化器init()方法,沒有重寫父類的指定初始化器init(age:), 所以該初始化器不能使用  error: incorrect argument label in call (have 'age:', expected 'no:')

  • required 修飾便捷初始化器
class Person {
    required convenience init() {
        self.init(age: 0)
    }
    init(age: Int) { }
}

class Student : Person {
    
    init(no: Int) {
        super.init(age: 0)
    }
    
    override init(age: Int) {
        super.init(age: age)
    }
    
    required convenience init() {
        self.init(age: 0)
    }
}

08-屬性觀察器

  • 父類的屬性在它自己的初始化器中賦值不會(huì)觸發(fā)屬性觀察器,但在子類的初始化器中賦值會(huì)觸發(fā)屬性觀察器
  • 父類的屬性在它自己的初始化器中賦值`不會(huì)觸發(fā)屬性觀察器
class Person {
    
    var age: Int {
        willSet {
            print("Person willSet", newValue)
        }
        didSet {
            print("Person didSet", oldValue)
        }
    }
    init() {
        self.age = 0    // 此處不會(huì)調(diào)用自己的屬性觀察器
    }
}
var p1 = Person()   // 此次沒有打印任何信息

  • 在子類的初始化器中賦值會(huì)觸發(fā)屬性觀察器`
class Person {
    
    var age: Int {
        willSet {
            print("Person willSet", newValue)
        }
        didSet {
            print("Person didSet", oldValue)
        }
    }
    init() {
        self.age = 0    // 此處不會(huì)調(diào)用自己的屬性觀察器
    }
}

class Student : Person {
    override init() {
        super.init()
        self.age = 1    // 此處會(huì)調(diào)用父類的屬性觀察器
    }
}
// Person willSet 1
// Person didSet 0
var stu1 = Student()

  • 子類和父類都實(shí)現(xiàn)父類存儲(chǔ)屬性的屬性觀察器, 則子類初始化器中對父類存儲(chǔ)屬性賦值時(shí),子類和父類的屬性觀察器都會(huì)調(diào)用
class Person {
    
    var age: Int {
        willSet {
            print("Person willSet", newValue)
        }
        didSet {
            print("Person didSet", oldValue)
        }
    }
    init() {
        self.age = 0    // 此處不會(huì)調(diào)用自己的屬性觀察器
    }
}

class Student : Person {
    
    override var age: Int {
        willSet {
            print("Student willSet", newValue)
        }
        didSet {
            print("Student didSet", oldValue)
        }
    }
    
    override init() {
        super.init()
        self.age = 1    // 此處會(huì)調(diào)用父類的屬性觀察器
    }
}
//Student willSet 1
//Person willSet 1
//Person didSet 0
//Student didSet 0
var stu1 = Student()

09-可失敗初始化器

  • 結(jié)構(gòu)體丙号、枚舉都可以使用init?定義可失敗初始化器
  • 不允許同時(shí)定義參數(shù)標(biāo)簽先朦、參數(shù)個(gè)數(shù)參數(shù)類型相同可失敗初始化器和非可失敗初始化器
  • 可以用init!定義隱式解包的可失敗初始化器
  • 可失敗初始化器可以調(diào)用非可失敗初始化器,非可失敗初始化器調(diào)用可失敗初始化器需要進(jìn)行解包
  • 如果初始化器調(diào)用一個(gè)可失敗初始化器導(dǎo)致初始化失敗,那么整個(gè)初始化過程都失敗,并且之后的代碼都停止執(zhí)行
  • 可以用一個(gè)非可失敗初始化器重寫一個(gè)可失敗初始化器,但反過來是不行的

class Person {
    var name: String
    init?(name: String) {   // 可選項(xiàng)初始化器 在init后面加上?
        if name.isEmpty {
            return nil
        }
        self.name = name
    }
}

var p1 = Person(name: "")
print(p1)   // nil

var p2 = Person(name: "Liwx")
print(p2)   // Optional(__lldb_expr_2.Person)

  • Int可失敗初始化器使用
// Int可失敗初始化器定義
// public Init?(_ description: String)
var num1 = Int("123")
print(num1) // Optional(123)
var num2 = Int("abc")
print(num2) // nil

  • 枚舉可失敗初始化器使用
enum Answer : Int {
    case wrong, right
}

var answer1 = Answer(rawValue: 1)
print(answer1)  // Optional(__lldb_expr_13.Answer.right)
var answer2 = Answer(rawValue: 100)
print(answer2)  // nil

  • 不允許同時(shí)定義參數(shù)標(biāo)簽犬缨、參數(shù)個(gè)數(shù)喳魏、參數(shù)類型相同的可失敗初始化器和非可失敗初始化器
class Person {
    var name: String
    init?(name: String) {   // 可選項(xiàng)初始化器 在init后面加上?
        if name.isEmpty {
            return nil
        }
        self.name = name
    }
    // 不允許同時(shí)定義參數(shù)標(biāo)簽、參數(shù)個(gè)數(shù)怀薛、參數(shù)類型相同的可失敗初始化器和非可失敗初始化器
//    init(name: String) {    // error: invalid redeclaration of 'init(name:)'
//        self.name = name
//    }
}

  • 可以用init!定義隱式解包的可失敗初始化器
class Person {
    var name: String
    init!(name: String) {   // 可選項(xiàng)初始化器 在init后面加上?
        if name.isEmpty {
            return nil
        }
        self.name = name
    }
}

var p1 = Person(name: "")
print(p1)   // nil

var p2 = Person(name: "Liwx")
print(p2)   // Optional(__lldb_expr_16.Person)

  • 可失敗初始化器可以調(diào)用非可失敗初始化器
class Person {
    var name: String
    convenience init?(name: String) {
        // 可失敗初始化器可以調(diào)用非可失敗初始化器
        self.init()
        
        if name.isEmpty {
            return nil
        }
    }
    
    init() {
        self.name = ""
    }
}

var p1 = Person(name: "")
print(p1)   // nil
var p2 = Person(name: "Liwx")
print(p2)   // Optional(__lldb_expr_19.Person)

  • 非可失敗初始化器調(diào)用可失敗初始化器需要進(jìn)行解包(慎用)
  • 這種情況調(diào)用,如果可失敗初始化器返回nil, 強(qiáng)制解包運(yùn)行時(shí)會(huì)崩潰
class Person {
    var name: String
    
    init?(name: String) {
        if name.isEmpty {
            return nil
        }
        self.name = name
    }

    convenience init() {
        // 非可失敗初始化器調(diào)用可失敗初始化器需要進(jìn)行解包
        self.init(name: "")!
    }
    
    // 這樣也可以
//    init!(name: String) {
//        if name.isEmpty {
//            return nil
//        }
//        self.name = name
//    }
//
//    convenience init() {
//        // 非可失敗初始化器調(diào)用可失敗初始化器需要進(jìn)行解包
//        self.init(name: "")
//    }
}
var p1 = Person()
print(p1)   // error: Unexpectedly found nil while unwrapping an Optional value

  • 如果初始化器調(diào)用一個(gè)可失敗初始化器導(dǎo)致初始化失敗,那么整個(gè)初始化過程都失敗,并且之后的代碼都停止執(zhí)行
class Person {
    var name: String
    
    init?(name: String) {
        if name.isEmpty {
            return nil
        }
        self.name = name
    }

    convenience init?() {
        // 非可失敗初始化器調(diào)用可失敗初始化器需要進(jìn)行解包
        self.init(name: "")
        // 可失敗初始化器如果初始化失敗,以下代碼不執(zhí)行
        self.name = "Liwx"
        print("test")   // 不會(huì)打印test
    }
}
var p1 = Person()
print(p1)   // nil

  • 可以用一個(gè)非可失敗初始化器 重寫一個(gè)可失敗初始化器,但反過來是不行
class Person {
    var name: String
    
    init?(name: String) {
        if name.isEmpty {
            return nil
        }
        self.name = name
    }
}

class Student : Person {
    // 可以用一個(gè)`非可失敗初始化器` `重寫`一個(gè)`可失敗初始化器`
    override init(name: String) {
        // 因?yàn)闆]有調(diào)用super,所以報(bào)錯(cuò) error: 'super.init' isn't called on all paths before returning from initializer
    }
}

  • 不可以用一個(gè)可失敗初始化器 重寫一個(gè)非可失敗初始化器
class Person {
    var name: String
    
    init(name: String) {
        self.name = name
    }
}

class Student : Person {
    // 不可以用一個(gè)`可失敗初始化器` `重寫`一個(gè)`非可失敗初始化器`
    // error: failable initializer 'init(name:)' cannot override a non-failable initializer
    override init?(name: String) {
        super.init(name: name)
    }
}

10-反初始化器(deinit)

  • deinit叫做反初始化器,類似于C++的析構(gòu)函數(shù)刺彩、OC中的dealloc方法
  • 當(dāng)類的實(shí)例對象被釋放內(nèi)存時(shí),就會(huì)調(diào)用實(shí)例對象的deinit方法
  • 結(jié)構(gòu)體枚舉不能使用deinit
class Person {
    deinit {
        print("Person deinit")  // 實(shí)例對象被釋放會(huì)調(diào)用deinit
    }
}
func test() {
    var p1 = Person()
}
print("1")  // 1
test()      // Person deinit
print("2")  // 2

  • deinit不接受任何參數(shù),不能寫小括號,不能自行調(diào)用
  • 父類的deinit能被子類繼承
  • 子類的deinit實(shí)現(xiàn)執(zhí)行完畢后會(huì)調(diào)用父類的deinit
class Person {
    deinit {
        print("Person deinit")
    }
}

class Student : Person {
    deinit {
        print("Student deinit")
    }
}
func test() {
    var stu1 = Student()
}

// 1
// Student deinit
// Person deinit
// 2
print("1")
test()
print("2")

iOS Swift 語法 底層原理內(nèi)存管理分析 專題:【iOS Swift5語法】

下一篇: 敬請期待
上一篇: 11 - 繼承


最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
禁止轉(zhuǎn)載,如需轉(zhuǎn)載請通過簡信或評論聯(lián)系作者。
  • 序言:七十年代末创倔,一起剝皮案震驚了整個(gè)濱河市嗡害,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌畦攘,老刑警劉巖霸妹,帶你破解...
    沈念sama閱讀 219,427評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異知押,居然都是意外死亡叹螟,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評論 3 395
  • 文/潘曉璐 我一進(jìn)店門台盯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來罢绽,“玉大人,你說我怎么就攤上這事爷恳∮欣拢” “怎么了?”我有些...
    開封第一講書人閱讀 165,747評論 0 356
  • 文/不壞的土叔 我叫張陵温亲,是天一觀的道長棚壁。 經(jīng)常有香客問我,道長栈虚,這世上最難降的妖魔是什么袖外? 我笑而不...
    開封第一講書人閱讀 58,939評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮魂务,結(jié)果婚禮上曼验,老公的妹妹穿的比我還像新娘。我一直安慰自己粘姜,他們只是感情好鬓照,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,955評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著孤紧,像睡著了一般豺裆。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上号显,一...
    開封第一講書人閱讀 51,737評論 1 305
  • 那天臭猜,我揣著相機(jī)與錄音,去河邊找鬼押蚤。 笑死蔑歌,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的揽碘。 我是一名探鬼主播次屠,決...
    沈念sama閱讀 40,448評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼园匹,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了帅矗?” 一聲冷哼從身側(cè)響起偎肃,我...
    開封第一講書人閱讀 39,352評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎浑此,沒想到半個(gè)月后累颂,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,834評論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡凛俱,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,992評論 3 338
  • 正文 我和宋清朗相戀三年紊馏,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蒲犬。...
    茶點(diǎn)故事閱讀 40,133評論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡朱监,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出原叮,到底是詐尸還是另有隱情赫编,我是刑警寧澤,帶...
    沈念sama閱讀 35,815評論 5 346
  • 正文 年R本政府宣布奋隶,位于F島的核電站擂送,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏唯欣。R本人自食惡果不足惜嘹吨,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,477評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望境氢。 院中可真熱鬧蟀拷,春花似錦、人聲如沸萍聊。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,022評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽寿桨。三九已至愈诚,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間牛隅,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,147評論 1 272
  • 我被黑心中介騙來泰國打工酌泰, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留媒佣,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,398評論 3 373
  • 正文 我出身青樓陵刹,卻偏偏與公主長得像默伍,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,077評論 2 355