欧美日韩亚-欧美日韩亚州在线-欧美日韩亚洲-欧美日韩亚洲第一区-欧美日韩亚洲二区在线-欧美日韩亚洲高清精品

金喜正规买球

logo Swift編程語(yǔ)言中文教程

文檔金喜正規(guī)買(mǎi)球>>Swift編程語(yǔ)言中文教程>>Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)


本頁(yè)包含內(nèi)容:

Swift 使用自動(dòng)引用計(jì)數(shù)(ARC)這一機(jī)制來(lái)跟蹤和管理你的應(yīng)用程序的內(nèi)存。通常情況下,Swift 的內(nèi)存管理機(jī)制會(huì)一直起著作用,你無(wú)須自己來(lái)考慮內(nèi)存的管理。ARC 會(huì)在類(lèi)的實(shí)例不再被使用時(shí),自動(dòng)釋放其占用的內(nèi)存。

然而,在少數(shù)情況下,ARC 為了能幫助你管理內(nèi)存,需要更多的關(guān)于你的代碼之間關(guān)系的信息。本章描述了這些情況,并且為你示范怎樣啟用 ARC 來(lái)管理你的應(yīng)用程序的內(nèi)存。

注意:

引用計(jì)數(shù)僅僅應(yīng)用于類(lèi)的實(shí)例。結(jié)構(gòu)體和枚舉類(lèi)型是值類(lèi)型,不是引用類(lèi)型,也不是通過(guò)引用的方式存儲(chǔ)和傳遞。

自動(dòng)引用計(jì)數(shù)的工作機(jī)制

當(dāng)你每次創(chuàng)建一個(gè)類(lèi)的新的實(shí)例的時(shí)候,ARC 會(huì)分配一大塊內(nèi)存用來(lái)儲(chǔ)存實(shí)例的信息。內(nèi)存中會(huì)包含實(shí)例的類(lèi)型信息,以及這個(gè)實(shí)例所有相關(guān)屬性的值。此外,當(dāng)實(shí)例不再被使用時(shí),ARC 釋放實(shí)例所占用的內(nèi)存,并讓釋放的內(nèi)存能挪作他用。這確保了不再被使用的實(shí)例,不會(huì)一直占用內(nèi)存空間。

然而,當(dāng) ARC 收回和釋放了正在被使用中的實(shí)例,該實(shí)例的屬性和方法將不能再被訪問(wèn)和調(diào)用。實(shí)際上,如果你試圖訪問(wèn)這個(gè)實(shí)例,你的應(yīng)用程序很可能會(huì)崩潰。

為了確保使用中的實(shí)例不會(huì)被銷(xiāo)毀,ARC 會(huì)跟蹤和計(jì)算每一個(gè)實(shí)例正在被多少屬性,常量和變量所引用。哪怕實(shí)例的引用數(shù)為一,ARC都不會(huì)銷(xiāo)毀這個(gè)實(shí)例。

為了使之成為可能,無(wú)論你將實(shí)例賦值給屬性,常量或者是變量,屬性,常量或者變量,都會(huì)對(duì)此實(shí)例創(chuàng)建強(qiáng)引用。之所以稱(chēng)之為強(qiáng)引用,是因?yàn)樗鼤?huì)將實(shí)例牢牢的保持住,只要強(qiáng)引用還在,實(shí)例是不允許被銷(xiāo)毀的。

自動(dòng)引用計(jì)數(shù)實(shí)踐

下面的例子展示了自動(dòng)引用計(jì)數(shù)的工作機(jī)制。例子以一個(gè)簡(jiǎn)單的Person類(lèi)開(kāi)始,并定義了一個(gè)叫name的常量屬性:

 class Person {
        let name: String

            init(name: String) {
            self.name = name
            println("\(name) is being initialized")
        }

            deinit {
            println("\(name) is being deinitialized")
        }
    }

Person類(lèi)有一個(gè)構(gòu)造函數(shù),此構(gòu)造函數(shù)為實(shí)例的name屬性賦值并打印出信息,以表明初始化過(guò)程生效。Person類(lèi)同時(shí)也擁有析構(gòu)函數(shù),同樣會(huì)在實(shí)例被銷(xiāo)毀的時(shí)候打印出信息。

接下來(lái)的代碼片段定義了三個(gè)類(lèi)型為Person?的變量,用來(lái)按照代碼片段中的順序,為新的Person實(shí)例建立多個(gè)引用。由于這些變量是被定義為可選類(lèi)型(Person?,而不是Person),它們的值會(huì)被自動(dòng)初始化為nil,目前還不會(huì)引用到Person類(lèi)的實(shí)例。

    var reference1: Person?
    var reference2: Person?
    var reference3: Person?

現(xiàn)在你可以創(chuàng)建Person類(lèi)的新實(shí)例,并且將它賦值給三個(gè)變量其中的一個(gè):

 reference1 = Person(name: "John Appleseed")
   // prints "John Appleseed is being initialized”

應(yīng)當(dāng)注意到當(dāng)你調(diào)用Person類(lèi)的構(gòu)造函數(shù)的時(shí)候,"John Appleseed is being initialized”會(huì)被打印出來(lái)。由此可以確定構(gòu)造函數(shù)被執(zhí)行。

由于Person類(lèi)的新實(shí)例被賦值給了reference1變量,所以reference1到Person類(lèi)的新實(shí)例之間建立了一個(gè)強(qiáng)引用。正是因?yàn)檫@個(gè)強(qiáng)引用,ARC 會(huì)保證Person實(shí)例被保持在內(nèi)存中不被銷(xiāo)毀。

如果你將同樣的Person實(shí)例也賦值給其他兩個(gè)變量,該實(shí)例又會(huì)多出兩個(gè)強(qiáng)引用:

    reference2 = reference1
    reference3 = reference1

現(xiàn)在這個(gè)Person實(shí)例已經(jīng)有三個(gè)強(qiáng)引用了。

如果你通過(guò)給兩個(gè)變量賦值nil的方式斷開(kāi)兩個(gè)強(qiáng)引用()包括最先的那個(gè)強(qiáng)引用),只留下一個(gè)強(qiáng)引用,Person實(shí)例不會(huì)被銷(xiāo)毀:

reference2 = nil
reference3 = nil

ARC 會(huì)在第三個(gè),也即最后一個(gè)強(qiáng)引用被斷開(kāi)的時(shí)候,銷(xiāo)毀Person實(shí)例,這也意味著你不再使用這個(gè)Person實(shí)例:

reference3 = nil
// prints "John Appleseed is being deinitialized"

類(lèi)實(shí)例之間的循環(huán)強(qiáng)引用

在上面的例子中,ARC 會(huì)跟蹤你所新創(chuàng)建的Person實(shí)例的引用數(shù)量,并且會(huì)在Person實(shí)例不再被需要時(shí)銷(xiāo)毀它。

然而,我們可能會(huì)寫(xiě)出這樣的代碼,一個(gè)類(lèi)永遠(yuǎn)不會(huì)有0個(gè)強(qiáng)引用。這種情況發(fā)生在兩個(gè)類(lèi)實(shí)例互相保持對(duì)方的強(qiáng)引用,并讓對(duì)方不被銷(xiāo)毀。這就是所謂的循環(huán)強(qiáng)引用。

你可以通過(guò)定義類(lèi)之間的關(guān)系為弱引用或者無(wú)主引用,以此替代強(qiáng)引用,從而解決循環(huán)強(qiáng)引用的問(wèn)題。具體的過(guò)程在解決類(lèi)實(shí)例之間的循環(huán)強(qiáng)引用中有描述。不管怎樣,在你學(xué)習(xí)怎樣解決循環(huán)強(qiáng)引用之前,很有必要了解一下它是怎樣產(chǎn)生的。

下面展示了一個(gè)不經(jīng)意產(chǎn)生循環(huán)強(qiáng)引用的例子。例子定義了兩個(gè)類(lèi):Person和Apartment,用來(lái)建模公寓和它其中的居民:

class Person {
    let name: String
    init(name: String) { self.name = name }
    var apartment: Apartment?
    deinit { println("\(name) is being deinitialized") }
}

class Apartment {
    let number: Int
    init(number: Int) { self.number = number }
    var tenant: Person?
    deinit { println("Apartment #\(number) is being deinitialized") }
}

每一個(gè)Person實(shí)例有一個(gè)類(lèi)型為String,名字為name的屬性,并有一個(gè)可選的初始化為nil的apartment屬性。apartment屬性是可選的,因?yàn)橐粋€(gè)人并不總是擁有公寓。

類(lèi)似的,每個(gè)Apartment實(shí)例有一個(gè)叫number,類(lèi)型為Int的屬性,并有一個(gè)可選的初始化為nil的tenant屬性。tenant屬性是可選的,因?yàn)橐粭澒⒉⒉豢偸怯芯用瘛?/p>

這兩個(gè)類(lèi)都定義了析構(gòu)函數(shù),用以在類(lèi)實(shí)例被析構(gòu)的時(shí)候輸出信息。這讓你能夠知曉Person和Apartment的實(shí)例是否像預(yù)期的那樣被銷(xiāo)毀。

接下來(lái)的代碼片段定義了兩個(gè)可選類(lèi)型的變量john和number73,并分別被設(shè)定為下面的Apartment和Person的實(shí)例。這兩個(gè)變量都被初始化為nil,并為可選的:

var john: Person?
var number73: Apartment?

現(xiàn)在你可以創(chuàng)建特定的Person和Apartment實(shí)例并將類(lèi)實(shí)例賦值給john和number73變量:

john = Person(name: "John Appleseed")
number73 = Apartment(number: 73)

在兩個(gè)實(shí)例被創(chuàng)建和賦值后,下圖表現(xiàn)了強(qiáng)引用的關(guān)系。變量john現(xiàn)在有一個(gè)指向Person實(shí)例的強(qiáng)引用,而變量number73有一個(gè)指向Apartment實(shí)例的強(qiáng)引用:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

現(xiàn)在你能夠?qū)⑦@兩個(gè)實(shí)例關(guān)聯(lián)在一起,這樣人就能有公寓住了,而公寓也有了房客。注意感嘆號(hào)是用來(lái)展開(kāi)和訪問(wèn)可選變量john和number73中的實(shí)例,這樣實(shí)例的屬性才能被賦值:

john!.apartment = number73
number73!.tenant = john

在將兩個(gè)實(shí)例聯(lián)系在一起之后,強(qiáng)引用的關(guān)系如圖所示:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

不幸的是,將這兩個(gè)實(shí)例關(guān)聯(lián)在一起之后,一個(gè)循環(huán)強(qiáng)引用被創(chuàng)建了。Person實(shí)例現(xiàn)在有了一個(gè)指向Apartment實(shí)例的強(qiáng)引用,而Apartment實(shí)例也有了一個(gè)指向Person實(shí)例的強(qiáng)引用。因此,當(dāng)你斷開(kāi)john和number73變量所持有的強(qiáng)引用時(shí),引用計(jì)數(shù)并不會(huì)降為 0,實(shí)例也不會(huì)被 ARC 銷(xiāo)毀:

john = nil
number73 = nil

注意,當(dāng)你把這兩個(gè)變量設(shè)為nil時(shí),沒(méi)有任何一個(gè)析構(gòu)函數(shù)被調(diào)用。強(qiáng)引用循環(huán)阻止了Person和Apartment類(lèi)實(shí)例的銷(xiāo)毀,并在你的應(yīng)用程序中造成了內(nèi)存泄漏。

在你將john和number73賦值為nil后,強(qiáng)引用關(guān)系如下圖:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

Person和Apartment實(shí)例之間的強(qiáng)引用關(guān)系保留了下來(lái)并且不會(huì)被斷開(kāi)。

解決實(shí)例之間的循環(huán)強(qiáng)引用

Swift 提供了兩種辦法用來(lái)解決你在使用類(lèi)的屬性時(shí)所遇到的循環(huán)強(qiáng)引用問(wèn)題:弱引用(weak reference)和無(wú)主引用(unowned reference)。

弱引用和無(wú)主引用允許循環(huán)引用中的一個(gè)實(shí)例引用另外一個(gè)實(shí)例而不保持強(qiáng)引用。這樣實(shí)例能夠互相引用而不產(chǎn)生循環(huán)強(qiáng)引用。

對(duì)于生命周期中會(huì)變?yōu)閚il的實(shí)例使用弱引用。相反的,對(duì)于初始化賦值后再也不會(huì)被賦值為nil的實(shí)例,使用無(wú)主引用。

弱引用

弱引用不會(huì)牢牢保持住引用的實(shí)例,并且不會(huì)阻止 ARC 銷(xiāo)毀被引用的實(shí)例。這種行為阻止了引用變?yōu)檠h(huán)強(qiáng)引用。聲明屬性或者變量時(shí),在前面加上weak關(guān)鍵字表明這是一個(gè)弱引用。

在實(shí)例的生命周期中,如果某些時(shí)候引用沒(méi)有值,那么弱引用可以阻止循環(huán)強(qiáng)引用。如果引用總是有值,則可以使用無(wú)主引用,在無(wú)主引用中有描述。在上面Apartment的例子中,一個(gè)公寓的生命周期中,有時(shí)是沒(méi)有“居民”的,因此適合使用弱引用來(lái)解決循環(huán)強(qiáng)引用。

注意: 弱引用必須被聲明為變量,表明其值能在運(yùn)行時(shí)被修改。弱引用不能被聲明為常量。

因?yàn)槿跻每梢詻](méi)有值,你必須將每一個(gè)弱引用聲明為可選類(lèi)型??蛇x類(lèi)型是在 Swift 語(yǔ)言中推薦的用來(lái)表示可能沒(méi)有值的類(lèi)型。

因?yàn)槿跻貌粫?huì)保持所引用的實(shí)例,即使引用存在,實(shí)例也有可能被銷(xiāo)毀。因此,ARC 會(huì)在引用的實(shí)例被銷(xiāo)毀后自動(dòng)將其賦值為nil。你可以像其他可選值一樣,檢查弱引用的值是否存在,你永遠(yuǎn)也不會(huì)遇到被銷(xiāo)毀了而不存在的實(shí)例。

下面的例子跟上面Person和Apartment的例子一致,但是有一個(gè)重要的區(qū)別。這一次,Apartment的tenant屬性被聲明為弱引用:

class Person {
    let name: String
    init(name: String) { self.name = name }
    var apartment: Apartment?
    deinit { println("\(name) is being deinitialized") }
}

class Apartment {
    let number: Int
    init(number: Int) { self.number = number }
    weak var tenant: Person?
    deinit { println("Apartment #\(number) is being deinitialized") }
}

然后跟之前一樣,建立兩個(gè)變量(john和number73)之間的強(qiáng)引用,并關(guān)聯(lián)兩個(gè)實(shí)例:

var john: Person?
var number73: Apartment?

john = Person(name: "John Appleseed")
number73 = Apartment(number: 73)

john!.apartment = number73
number73!.tenant = john

現(xiàn)在,兩個(gè)關(guān)聯(lián)在一起的實(shí)例的引用關(guān)系如下圖所示:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

Person實(shí)例依然保持對(duì)Apartment實(shí)例的強(qiáng)引用,但是Apartment實(shí)例只是對(duì)Person實(shí)例的弱引用。這意味著當(dāng)你斷開(kāi)john變量所保持的強(qiáng)引用時(shí),再也沒(méi)有指向Person實(shí)例的強(qiáng)引用了:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

由于再也沒(méi)有指向Person實(shí)例的強(qiáng)引用,該實(shí)例會(huì)被銷(xiāo)毀:

john = nil
// prints "John Appleseed is being deinitialized"

唯一剩下的指向Apartment實(shí)例的強(qiáng)引用來(lái)自于變量number73。如果你斷開(kāi)這個(gè)強(qiáng)引用,再也沒(méi)有指向Apartment實(shí)例的強(qiáng)引用了:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

由于再也沒(méi)有指向Apartment實(shí)例的強(qiáng)引用,該實(shí)例也會(huì)被銷(xiāo)毀:

number73 = nil
// prints "Apartment #73 is being deinitialized"

上面的兩段代碼展示了變量john和number73在被賦值為nil后,Person實(shí)例和Apartment實(shí)例的析構(gòu)函數(shù)都打印出“銷(xiāo)毀”的信息。這證明了引用循環(huán)被打破了。

無(wú)主引用

和弱引用類(lèi)似,無(wú)主引用不會(huì)牢牢保持住引用的實(shí)例。和弱引用不同的是,無(wú)主引用是永遠(yuǎn)有值的。因此,無(wú)主引用總是被定義為非可選類(lèi)型(non-optional type)。你可以在聲明屬性或者變量時(shí),在前面加上關(guān)鍵字unowned表示這是一個(gè)無(wú)主引用。

由于無(wú)主引用是非可選類(lèi)型,你不需要在使用它的時(shí)候?qū)⑺归_(kāi)。無(wú)主引用總是可以被直接訪問(wèn)。不過(guò) ARC 無(wú)法在實(shí)例被銷(xiāo)毀后將無(wú)主引用設(shè)為nil,因?yàn)榉强蛇x類(lèi)型的變量不允許被賦值為nil。

注意: 如果你試圖在實(shí)例被銷(xiāo)毀后,訪問(wèn)該實(shí)例的無(wú)主引用,會(huì)觸發(fā)運(yùn)行時(shí)錯(cuò)誤。使用無(wú)主引用,你必須確保引用始終指向一個(gè)未銷(xiāo)毀的實(shí)例。

還需要注意的是如果你試圖訪問(wèn)實(shí)例已經(jīng)被銷(xiāo)毀的無(wú)主引用,程序會(huì)直接崩潰,而不會(huì)發(fā)生無(wú)法預(yù)期的行為。所以你應(yīng)當(dāng)避免這樣的事情發(fā)生。

下面的例子定義了兩個(gè)類(lèi),Customer和CreditCard,模擬了銀行客戶(hù)和客戶(hù)的信用卡。這兩個(gè)類(lèi)中,每一個(gè)都將另外一個(gè)類(lèi)的實(shí)例作為自身的屬性。這種關(guān)系會(huì)潛在的創(chuàng)造循環(huán)強(qiáng)引用。

Customer和CreditCard之間的關(guān)系與前面弱引用例子中Apartment和Person的關(guān)系截然不同。在這個(gè)數(shù)據(jù)模型中,一個(gè)客戶(hù)可能有或者沒(méi)有信用卡,但是一張信用卡總是關(guān)聯(lián)著一個(gè)客戶(hù)。為了表示這種關(guān)系,Customer類(lèi)有一個(gè)可選類(lèi)型的card屬性,但是CreditCard類(lèi)有一個(gè)非可選類(lèi)型的customer屬性。

此外,只能通過(guò)將一個(gè)number值和customer實(shí)例傳遞給CreditCard構(gòu)造函數(shù)的方式來(lái)創(chuàng)建CreditCard實(shí)例。這樣可以確保當(dāng)創(chuàng)建CreditCard實(shí)例時(shí)總是有一個(gè)customer實(shí)例與之關(guān)聯(lián)。

由于信用卡總是關(guān)聯(lián)著一個(gè)客戶(hù),因此將customer屬性定義為無(wú)主引用,用以避免循環(huán)強(qiáng)引用:

class Customer {
    let name: String
    var card: CreditCard?
    init(name: String) {
        self.name = name
    }
    deinit { println("\(name) is being deinitialized") }
}

class CreditCard {
    let number: Int
    unowned let customer: Customer
    init(number: Int, customer: Customer) {
        self.number = number
        self.customer = customer
    }
    deinit { println("Card #\(number) is being deinitialized") }
}

下面的代碼片段定義了一個(gè)叫john的可選類(lèi)型Customer變量,用來(lái)保存某個(gè)特定客戶(hù)的引用。由于是可選類(lèi)型,所以變量被初始化為nil。

var john: Customer?

現(xiàn)在你可以創(chuàng)建Customer類(lèi)的實(shí)例,用它初始化CreditCard實(shí)例,并將新創(chuàng)建的CreditCard實(shí)例賦值為客戶(hù)的card屬性。

john = Customer(name: "John Appleseed")
john!.card = CreditCard(number: 1234_5678_9012_3456, customer: john!)

在你關(guān)聯(lián)兩個(gè)實(shí)例后,他們的引用關(guān)系如下圖所示:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

Customer實(shí)例持有對(duì)CreditCard實(shí)例的強(qiáng)引用,而CreditCard實(shí)例持有對(duì)Customer實(shí)例的無(wú)主引用。

由于customer的無(wú)主引用,當(dāng)你斷開(kāi)john變量持有的強(qiáng)引用時(shí),再也沒(méi)有指向Customer實(shí)例的強(qiáng)引用了:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

由于再也沒(méi)有指向Customer實(shí)例的強(qiáng)引用,該實(shí)例被銷(xiāo)毀了。其后,再也沒(méi)有指向CreditCard實(shí)例的強(qiáng)引用,該實(shí)例也隨之被銷(xiāo)毀了:

john = nil
// prints "John Appleseed is being deinitialized"
// prints "Card #1234567890123456 is being deinitialized"

最后的代碼展示了在john變量被設(shè)為nil后Customer實(shí)例和CreditCard實(shí)例的構(gòu)造函數(shù)都打印出了“銷(xiāo)毀”的信息。

無(wú)主引用以及隱式解析可選屬性

上面弱引用和無(wú)主引用的例子涵蓋了兩種常用的需要打破循環(huán)強(qiáng)引用的場(chǎng)景。

Person和Apartment的例子展示了兩個(gè)屬性的值都允許為nil,并會(huì)潛在的產(chǎn)生循環(huán)強(qiáng)引用。這種場(chǎng)景最適合用弱引用來(lái)解決。

Customer和CreditCard的例子展示了一個(gè)屬性的值允許為nil,而另一個(gè)屬性的值不允許為nil,并會(huì)潛在的產(chǎn)生循環(huán)強(qiáng)引用。這種場(chǎng)景最適合通過(guò)無(wú)主引用來(lái)解決。

然而,存在著第三種場(chǎng)景,在這種場(chǎng)景中,兩個(gè)屬性都必須有值,并且初始化完成后不能為nil。在這種場(chǎng)景中,需要一個(gè)類(lèi)使用無(wú)主屬性,而另外一個(gè)類(lèi)使用隱式解析可選屬性。

這使兩個(gè)屬性在初始化完成后能被直接訪問(wèn)(不需要可選展開(kāi)),同時(shí)避免了循環(huán)引用。這一節(jié)將為你展示如何建立這種關(guān)系。

下面的例子定義了兩個(gè)類(lèi),Country和City,每個(gè)類(lèi)將另外一個(gè)類(lèi)的實(shí)例保存為屬性。在這個(gè)模型中,每個(gè)國(guó)家必須有首都,而每一個(gè)城市必須屬于一個(gè)國(guó)家。為了實(shí)現(xiàn)這種關(guān)系,Country類(lèi)擁有一個(gè)capitalCity屬性,而City類(lèi)有一個(gè)country屬性:

class Country {
    let name: String
    let capitalCity: City!
    init(name: String, capitalName: String) {
        self.name = name
        self.capitalCity = City(name: capitalName, country: self)
    }
}

class City {
    let name: String
    unowned let country: Country
    init(name: String, country: Country) {
        self.name = name
        self.country = country
    }
}

為了建立兩個(gè)類(lèi)的依賴(lài)關(guān)系,City的構(gòu)造函數(shù)有一個(gè)Country實(shí)例的參數(shù),并且將實(shí)例保存為country屬性。

Country的構(gòu)造函數(shù)調(diào)用了City的構(gòu)造函數(shù)。然而,只有Country的實(shí)例完全初始化完后,Country的構(gòu)造函數(shù)才能把self傳給City的構(gòu)造函數(shù)。(在兩段式構(gòu)造過(guò)程中有具體描述)

為了滿(mǎn)足這種需求,通過(guò)在類(lèi)型結(jié)尾處加上感嘆號(hào)(City!)的方式,將Country的capitalCity屬性聲明為隱式解析可選類(lèi)型的屬性。這表示像其他可選類(lèi)型一樣,capitalCity屬性的默認(rèn)值為nil,但是不需要展開(kāi)他的值就能訪問(wèn)它。(在隱式解析可選類(lèi)型中有描述)

由于capitalCity默認(rèn)值為nil,一旦Country的實(shí)例在構(gòu)造函數(shù)中給name屬性賦值后,整個(gè)初始化過(guò)程就完成了。這代表一旦name屬性被賦值后,Country的構(gòu)造函數(shù)就能引用并傳遞隱式的self。Country的構(gòu)造函數(shù)在賦值capitalCity時(shí),就能將self作為參數(shù)傳遞給City的構(gòu)造函數(shù)。

以上的意義在于你可以通過(guò)一條語(yǔ)句同時(shí)創(chuàng)建Country和City的實(shí)例,而不產(chǎn)生循環(huán)強(qiáng)引用,并且capitalCity的屬性能被直接訪問(wèn),而不需要通過(guò)感嘆號(hào)來(lái)展開(kāi)它的可選值:

var country = Country(name: "Canada", capitalName: "Ottawa")
println("\(country.name)'s capital city is called \(country.capitalCity.name)")
// prints "Canada's capital city is called Ottawa"

在上面的例子中,使用隱式解析可選值的意義在于滿(mǎn)足了兩個(gè)類(lèi)構(gòu)造函數(shù)的需求。capitalCity屬性在初始化完成后,能像非可選值一樣使用和存取同時(shí)還避免了循環(huán)強(qiáng)引用。

閉包引起的循環(huán)強(qiáng)引用

前面我們看到了循環(huán)強(qiáng)引用環(huán)是在兩個(gè)類(lèi)實(shí)例屬性互相保持對(duì)方的強(qiáng)引用時(shí)產(chǎn)生的,還知道了如何用弱引用和無(wú)主引用來(lái)打破循環(huán)強(qiáng)引用。

循環(huán)強(qiáng)引用還會(huì)發(fā)生在當(dāng)你將一個(gè)閉包賦值給類(lèi)實(shí)例的某個(gè)屬性,并且這個(gè)閉包體中又使用了實(shí)例。這個(gè)閉包體中可能訪問(wèn)了實(shí)例的某個(gè)屬性,例如self.someProperty,或者閉包中調(diào)用了實(shí)例的某個(gè)方法,例如self.someMethod。這兩種情況都導(dǎo)致了閉包 “捕獲" self,從而產(chǎn)生了循環(huán)強(qiáng)引用。

循環(huán)強(qiáng)引用的產(chǎn)生,是因?yàn)殚]包和類(lèi)相似,都是引用類(lèi)型。當(dāng)你把一個(gè)閉包賦值給某個(gè)屬性時(shí),你也把一個(gè)引用賦值給了這個(gè)閉包。實(shí)質(zhì)上,這跟之前的問(wèn)題是一樣的-兩個(gè)強(qiáng)引用讓彼此一直有效。但是,和兩個(gè)類(lèi)實(shí)例不同,這次一個(gè)是類(lèi)實(shí)例,另一個(gè)是閉包。

Swift 提供了一種優(yōu)雅的方法來(lái)解決這個(gè)問(wèn)題,稱(chēng)之為閉包占用列表(closuer capture list)。同樣的,在學(xué)習(xí)如何用閉包占用列表破壞循環(huán)強(qiáng)引用之前,先來(lái)了解一下循環(huán)強(qiáng)引用是如何產(chǎn)生的,這對(duì)我們是很有幫助的。

下面的例子為你展示了當(dāng)一個(gè)閉包引用了self后是如何產(chǎn)生一個(gè)循環(huán)強(qiáng)引用的。例子中定義了一個(gè)叫HTMLElement的類(lèi),用一種簡(jiǎn)單的模型表示 HTML 中的一個(gè)單獨(dú)的元素:

class HTMLElement {

    let name: String
    let text: String?

    @lazy var asHTML: () -> String = {
        if let text = self.text {
            return "<\(self.name)>\(text)</\(self.name)>"
        } else {
            return "<\(self.name) />"
        }
    }

    init(name: String, text: String? = nil) {
        self.name = name
        self.text = text
    }

    deinit {
        println("\(name) is being deinitialized")
    }

}

HTMLElement類(lèi)定義了一個(gè)name屬性來(lái)表示這個(gè)元素的名稱(chēng),例如代表段落的"p",或者代表?yè)Q行的"br"。HTMLElement還定義了一個(gè)可選屬性text,用來(lái)設(shè)置和展現(xiàn) HTML 元素的文本。

除了上面的兩個(gè)屬性,HTMLElement還定義了一個(gè)lazy屬性asHTML。這個(gè)屬性引用了一個(gè)閉包,將name和text組合成 HTML 字符串片段。該屬性是() -> String類(lèi)型,或者可以理解為“一個(gè)沒(méi)有參數(shù),返回String的函數(shù)”。

默認(rèn)情況下,閉包賦值給了asHTML屬性,這個(gè)閉包返回一個(gè)代表 HTML 標(biāo)簽的字符串。如果text值存在,該標(biāo)簽就包含可選值text;如果text不存在,該標(biāo)簽就不包含文本。對(duì)于段落元素,根據(jù)text是"some text"還是nil,閉包會(huì)返回

<p>some text</p>

或者

可以像實(shí)例方法那樣去命名、使用asHTML屬性。然而,由于asHTML是閉包而不是實(shí)例方法,如果你想改變特定元素的 HTML 處理的話(huà),可以用自定義的閉包來(lái)取代默認(rèn)值。

注意:

asHTML聲明為lazy屬性,因?yàn)橹挥挟?dāng)元素確實(shí)需要處理為HTML輸出的字符串時(shí),才需要使用asHTML。也就是說(shuō),在默認(rèn)的閉包中可以使用self,因?yàn)橹挥挟?dāng)初始化完成以及self確實(shí)存在后,才能訪問(wèn)lazy屬性。

HTMLElement類(lèi)只提供一個(gè)構(gòu)造函數(shù),通過(guò)name和text(如果有的話(huà))參數(shù)來(lái)初始化一個(gè)元素。該類(lèi)也定義了一個(gè)析構(gòu)函數(shù),當(dāng)HTMLElement實(shí)例被銷(xiāo)毀時(shí),打印一條消息。

下面的代碼展示了如何用HTMLElement類(lèi)創(chuàng)建實(shí)例并打印消息。

var paragraph: HTMLElement? = HTMLElement(name: "p", text: "hello, world")
println(paragraph!.asHTML())
// prints"hello, world"

注意:

上面的paragraph變量定義為可選HTMLElement,因此我們可以賦值nil給它來(lái)演示循環(huán)強(qiáng)引用。

不幸的是,上面寫(xiě)的HTMLElement類(lèi)產(chǎn)生了類(lèi)實(shí)例和asHTML默認(rèn)值的閉包之間的循環(huán)強(qiáng)引用。循環(huán)強(qiáng)引用如下圖所示:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

實(shí)例的asHTML屬性持有閉包的強(qiáng)引用。但是,閉包在其閉包體內(nèi)使用了self(引用了self.name和self.text),因此閉包捕獲了self,這意味著閉包又反過(guò)來(lái)持有了HTMLElement實(shí)例的強(qiáng)引用。這樣兩個(gè)對(duì)象就產(chǎn)生了循環(huán)強(qiáng)引用。(更多關(guān)于閉包捕獲值的信息,請(qǐng)參考值捕獲)。

注意:

雖然閉包多次使用了self,它只捕獲HTMLElement實(shí)例的一個(gè)強(qiáng)引用。

如果設(shè)置paragraph變量為nil,打破它持有的HTMLElement實(shí)例的強(qiáng)引用,HTMLElement實(shí)例和它的閉包都不會(huì)被銷(xiāo)毀,也是因?yàn)檠h(huán)強(qiáng)引用:

paragraph = nil

注意HTMLElementdeinitializer中的消息并沒(méi)有別打印,證明了HTMLElement實(shí)例并沒(méi)有被銷(xiāo)毀。

解決閉包引起的循環(huán)強(qiáng)引用

在定義閉包時(shí)同時(shí)定義捕獲列表作為閉包的一部分,通過(guò)這種方式可以解決閉包和類(lèi)實(shí)例之間的循環(huán)強(qiáng)引用。捕獲列表定義了閉包體內(nèi)捕獲一個(gè)或者多個(gè)引用類(lèi)型的規(guī)則。跟解決兩個(gè)類(lèi)實(shí)例間的循環(huán)強(qiáng)引用一樣,聲明每個(gè)捕獲的引用為弱引用或無(wú)主引用,而不是強(qiáng)引用。應(yīng)當(dāng)根據(jù)代碼關(guān)系來(lái)決定使用弱引用還是無(wú)主引用。

注意:

Swift 有如下要求:只要在閉包內(nèi)使用self的成員,就要用self.someProperty或者self.someMethod(而不只是someProperty或someMethod)。這提醒你可能會(huì)不小心就捕獲了self。

定義捕獲列表

捕獲列表中的每個(gè)元素都是由weak或者unowned關(guān)鍵字和實(shí)例的引用(如self或someInstance)成對(duì)組成。每一對(duì)都在方括號(hào)中,通過(guò)逗號(hào)分開(kāi)。

捕獲列表放置在閉包參數(shù)列表和返回類(lèi)型之前:

@lazy var someClosure: (Int, String) -> String = {
    [unowned self] (index: Int, stringToProcess: String) -> String in
    // closure body goes here
}

如果閉包沒(méi)有指定參數(shù)列表或者返回類(lèi)型,則可以通過(guò)上下文推斷,那么可以捕獲列表放在閉包開(kāi)始的地方,跟著是關(guān)鍵字in:

@lazy var someClosure: () -> String = {
    [unowned self] in
    // closure body goes here
}

弱引用和無(wú)主引用

當(dāng)閉包和捕獲的實(shí)例總是互相引用時(shí)并且總是同時(shí)銷(xiāo)毀時(shí),將閉包內(nèi)的捕獲定義為無(wú)主引用。

相反的,當(dāng)捕獲引用有時(shí)可能會(huì)是nil時(shí),將閉包內(nèi)的捕獲定義為弱引用。弱引用總是可選類(lèi)型,并且當(dāng)引用的實(shí)例被銷(xiāo)毀后,弱引用的值會(huì)自動(dòng)置為nil。這使我們可以在閉包內(nèi)檢查他們是否存在。

注意:

如果捕獲的引用絕對(duì)不會(huì)置為nil,應(yīng)該用無(wú)主引用,而不是弱引用。

前面的HTMLElement例子中,無(wú)主引用是正確的解決循環(huán)強(qiáng)引用的方法。這樣編寫(xiě)HTMLElement類(lèi)來(lái)避免循環(huán)強(qiáng)引用:

class HTMLElement {

    let name: String
    let text: String?

    @lazy var asHTML: () -> String = {
        [unowned self] in
        if let text = self.text {
            return "<\(self.name)>\(text)</\(self.name)>"
        } else {
            return "<\(self.name) />"
        }
    }

    init(name: String, text: String? = nil) {
        self.name = name
        self.text = text
    }

    deinit {
        println("\(name) is being deinitialized")
    }

}

上面的HTMLElement實(shí)現(xiàn)和之前的實(shí)現(xiàn)一致,只是在asHTML閉包中多了一個(gè)捕獲列表。這里,捕獲列表是[unowned self],表示“用無(wú)主引用而不是強(qiáng)引用來(lái)捕獲self”。

和之前一樣,我們可以創(chuàng)建并打印HTMLElement實(shí)例:

var paragraph: HTMLElement? = HTMLElement(name: "p", text: "hello, world")
println(paragraph!.asHTML())
// prints "<p>hello, world</p>"

使用捕獲列表后引用關(guān)系如下圖所示:

Swift編程語(yǔ)言中文教程(十六):Swift自動(dòng)引用計(jì)數(shù)

這一次,閉包以無(wú)主引用的形式捕獲self,并不會(huì)持有HTMLElement實(shí)例的強(qiáng)引用。如果將paragraph賦值為nil,HTMLElement實(shí)例將會(huì)被銷(xiāo)毀,并能看到它的析構(gòu)函數(shù)打印出的消息。

paragraph = nil
// prints "p is being deinitialized"
掃碼咨詢(xún)


添加微信 立即咨詢(xún)

電話(huà)咨詢(xún)

客服熱線(xiàn)
023-68661681

TOP
一级一黄免费视频在线看 | 91日本在线观看亚洲精品 | 亚洲aⅴ无| 加勒比精品| 国产不卡在线观看视频 | 91九色在线观看 | 国产自在线观看免费视频 | 蜜桃mv在线播放免费观看视频 | 国产亚洲精品综合一区 | 2025国产精品极品色在线 | 日本一区二区三区四区在线 | 国产美女在线精品免费观看 | 亚洲综合色区另类aⅴ | 女日韩优在线 | 日韩亚洲人成在线综合日本 | 国产成a人片在线观看视频下载 | 日本日本乱码伦视频在线观 | 重口视频二区在线观看 | 人成视频在线观看 | 日本免费精品 | 亚洲精品综合在线影院 | 国产精品多人 | 亚洲人精品 | 国产精品进线69影院在线 | 亚洲人成图片小说网站 | 四库国产精品成人 | 亚洲黄免费看网站 | 午夜未满十八勿入网站2 | 国产极品一区 | 污视频在线观看国产的 | 久中文字幕中文字幕亚洲无线 | 99re热这里只 | 国产美女一级做视须爱 | 日本毛x片免费视频观看视频 | 亚洲欧美综合乱伦一区 | 豆奶app官方网站 | 午夜电影网首页 | 好硬好大 | 另类专区亚洲 | 国产精品午夜小视频观看 | 国产ch| 中文字幕一区在线观看视频 | 免费电影天堂 | 国产伦精品 | 在线亚洲一区二区 | 亚洲影视日本欧美 | 日韩一区二区视频在线观看 | 亚洲国产一区二区在线观看 | 日本乱码一区二 | 精品国产欧美一区二区三区成人 | 亚洲色成人影院在线观看 | 日本有码中文字幕第一页在线播放 | 91大视频网站 | 国产亚洲精品第一综合另类 | 国语高清精品 | 欧亚天堂在线播放 | 永久免费的污视频网站 | 国产一区二区三区高清在线观看 | 国产一区二区三区欧美亚洲 | 国产在线观看精品 | 亚洲中文字幕乱码一区 | 绯色一区二区 | 一区二区三区免费在线视频 | 国产一区二区三区四区在 | 国产亚洲中文一区二区三区 | 加勒比综合精品 | 日本一区二区三区不卡视频中 | 看看永久 | 精品欧美日韩在线视频 | 人性情感短 | 国产日本欧美亚洲精品视 | 亚洲综合一区三区 | 中文字幕亚洲欧美一区 | 妖精视频一区二区免费 | 国产精品丝袜在线观看首页 | 国产精品+日 | 国产suv精品一区二区 | 激情欧美日韩一 | a∨中文字幕另类 | 欧美丝袜自拍制服另类 | 狠狠影院 | 亚洲成色综 | 91尤物国产 | 在线看片免费人成视频福利 | 国产日韩在线视看高清视频手机 | 日韩a级片在线观看 | 久热爱精品视频在线 | 二区三区在线播放 | 欧美一级高清视频在线播放 | 日韩中文字葛高清在线专区 | 精品亚洲一区二区三区在线播放 | 亚洲欧洲精品一二三区 | 91国偷自产中文字幕婷婷 | 色片在线观看 | 中文字幕久热精品 | 国产美女视频国产视视频 | 国产一区二区三区免费在线观看 | 日本国产性爱观看视频 | 欧美乱妇高清免费96欧美乱妇高 | 日本成a人片在线观看网址 国产精品蜜桃丝袜 | 国产老熟女精品视 | 国产小视频在线观看免费 | 亚洲国产精品18 | 日韩精品视频 | 色综合天天综合网国产国产人 | 国产精品福利自产拍在线观看 | 国产精品多p对白交换绿 | 精品国产欧美在线小说区 | 亚洲va国产日韩欧美精品 | 色老大综合 | 国产馆精品推荐在线观看 | 国产亚洲男人的天堂在线观看 | 精品熟人 | 日韩欧美综合一区二区三区 | 都市激情第一页 | 国产欧美日韩综合精品区一区二区 | 国产精品播放一区二区三区 | 最近中文字幕无吗高清免费视频 | 一边喂奶一边被爱 | 国产在线欧美观看 | 国产午夜福利一区二区三区在 | 久9视频这里只有精品试看 经典影片免费在线观看 | 久99视频精品免费观看福利 | 欧美人与动牲 | 91视频直播 | 99偷拍视频精品一区二区 | 国产欧美一区二区精品婷婷 | 天堂亚洲国产日韩在线看 | 国内一区二区三区在线观看 | 人摸人摸在线视频 | 国产精品专区 | 国产日韩综合在线视频 | a天堂中文在线天堂资源中文 | 日本黄页 | 欧美日韩在线精品一区二区三区 | 国产精品系列专区 | 国产性猛交╳xxx乱大交 | 欧美精品亚洲 | 国产在线视频欧美亚综合 | 亚洲国产综合人成综合网站 | 欧美激情片区一区二区三区 | 欧美激情000ⅹxx同性 | 国产做a∨在线视频 | 国产精品自产拍在线观看网站 | 精品国产高清自在线一区二区 | 日本最新在线一区二区 | 亚洲国产日韩在线观看 | 电视剧大全免费在线观看 | 国产一区不卡在线观看 | 国产精选在线观看播放 | 亚洲国产一区二区在线 | 中日韩无砖码一线二线 | 高清欧美日韩一区二区三区在线 | 国产九九自拍电影在线观看 | 97在线视频观看在线观看视频 | 2025免费高清电影大全 | 国产亚洲综合一区柠檬导航 | 日本一区二区三区免费播放视频站 | 国产厨房一区二区三区 | 欧洲乱码| 成a人片在线观看手机看 | 成人午夜影院 | 最新福利电影在线看 | 国产精品精品国 | 精品亚洲一区二区在线播放 | 国产精品日本一区二区不卡视频 | 最近中文字幕 | 欧美三级在线观看国产 | 国产在线拍精品热 | 国产精品秘果冻 | 免费高清欧美一区二区三区 | 日本一区二区日本免费 | 成人午夜影院网站 | 91蜜桃 | 日本三级香港三级人妇99 | 中文字幕在线免费 | 国产真实 | 日韩欧美亚洲一区二区在线观看 | 九九视频在线观看视频6 | 国产欧美| 特黄特色的大片观看免费视 | 又粗又大又黄又硬高清视频 | 欧美日韩一区二区三区视频网站 | 婷婷亚洲综合一区二区 | 日本a优不卡在线播放 | 美国十次啦超级大导航 | 不卡中文字幕系 | 国产又黄又猛又粗又爽的 | 日本三级韩国三级欧美三级 | 亚洲日韩欧美国产高清αv 亚洲欧美 | 欧美日韩在线免费观看 | 性生生活12分钟免费 | 黑人精品一区二区三区不卡 | 国精产品一区一区三区mba下载 | 精品国产日韩无影视 | 国产精品系列专区 | 国产精品专区第5页 | 午夜成人性刺激免费视频在线观看 | 丰满尤物一区二区三区 | 午夜视频在线免费观看 | 日韩成人一区二区三区在线观看 | 欧美国产另 | 国产精品一区在线免费观看 | 日韩精品中文字幕在线 | 97人人超| 中国国产免费毛卡片 | 国产91精品老熟女泻火 | a级高清观看视频在线看 | 欧美精品国产制服一区 | 一区二区三区四区在线不卡高清 | 日韩男女激情视频在线观看 | 日本一道在线播放高清 | 国产福利在线网址成人 | 日韩一区二区三区免费精品 | 在线观看成人影院 | 正在播放国产多p交换视频 日韩成全视频观看免费观看高清 | 亚洲国产系列一区二区三区 | 一区二区视频在线观看 | 欧美综合在线观看日本 | 天天躁恨恨躁夜躁2025 | 国产精品综合一区二区三区 | 五十路息与子在线观看 | 亚洲精品国产字幕 | 国产精品v | 欧美亚洲日韩国 | 中文字幕自拍偷拍 | 日本最新在线一区二区 | 美女视频黄频a | 国产人妇三级视频在线观看 | 丝袜视频 | 亚洲制服| 国产又滑又嫩又白 | 海角社真实xxⅹ人伦 | 国产精品不卡一区二区 | 国产a级三级三区成人国产一级婬 | 国产高清在线视频伊甸园 | 天天看片高清观看免费国产 | 日韩一区高清在线观看 | 国产一级a毛一级a看免费视频 | 国产美女高清片免费观看 | 国产系列在线精品二区 | 真实国产乱子伦精品 | 爽了在线视频 | 日本肥老妇色xxxxx日本老妇 | 丁香婷婷六月综合缴清 | 国内揄拍 | 日本精品一区二区在线播放 | 丰满岳乱妇一区二区三区 | 国精产品一区二区三区有限 | 午夜免费福利体验 | 国产精品后| 欧美乱妇高清无乱码在线观看 | 欧美人成网站观看www | 欧美精品在线播放 | 免费电影网站在线观看 | 经典影片免费在线观看 | 日韩一区二区三区免费视 | 国产女技师按摩在线观看 | 色猫咪免费人成网站在线观看 | 国产精品一品二区三区的使用体验 | 国产免费艾 | 成人免费视频软件网站 | 中日韩精品视频 | 国产伦理一区的二区三区四区 | 91国内揄拍国内精品情侣对白 | 亚洲精品一区二区三区人妖 | 成人日韩在线视频观看 | 欧美精品亚洲精品日韩专区v | 国产视频99kai | 日本高清一区二区 | 性欧美极品xxxx欧美一 | 在线视频精品免费 | 欧洲亚洲精品 | 草草精品在线视频 | 亚洲影院 | 亚洲中文字幕无线 | 亚洲欧美另类在线观看一区二区 | 国产在线精品成人一区二区三区 | 成人永久免费视频网站 | 国产熟女熟女 | 又刺激又爽又黄的视频在线观看 | 国产亚洲日韩欧 | 91精品欧美一区二区综合在线 | 免费岛国 | 国产午夜精华2025在线 | 日韩欧美亚洲每日更新在线观看 | 91精品国产| 国产精品天干天干在线综合 | 96精品专区国产在线观看高清 | 国精品无| 乌克兰18极品xx00喷水 | 人在线观看 | 日本三级韩国三级香港三级a级 | 国内自拍亚洲 | 国产亚洲新免费视 | 国语自产偷成人精品视频 | 国产免费高清视频在线观看不卡 | 国产98色在线| 亚洲国产网站在线观看 | 国产欧美精品一区二区三区pp | 女被男啪到哭的视频网站 | 国产亚洲一区二区三区日本 | 区三区国产高清视频 | 亚洲中文字幕高清有码在线 | 天天射寡妇射 | 44极品视频在 | 欧美区一区| 国产第一在线视频 | 美女被肏翻白眼视频在线观看 | 免费a级毛 | 日本黄页网址在线看免费不卡 | 日韩a在线播放 | 免费黃色三級片在线观看18 | 亚洲国产一区二区三区亚瑟 | 天天夜碰日日 | 亚洲欧洲日韩一区二区日本 | 一扒二脱三插片在 | 99热门精品一区二区三区无 | 日本一卡2卡3卡无卡免费 | 神马影院首页 | 国产小视频在线观看免费 | 国产在线 | 老少配老妇老熟女中文 | 一本一道日韩一二三四区免费 | 国产免费观看青青草原网站 | 国语自产偷拍精品视频偷 | 一区二区三区在线观看欧美日韩 | 亚洲人成在线观看影院 | 国产精品精品国 | 国产欧美视频一区二区三区 | 成人免费观看视频 | 亚洲精品影院在 | 亚洲日韩在线观看免费视频 | 资源视频在线观看 | 美女裸身网站免费看免费网站 | 午夜亚洲国产理论片二 | 国产又粗又猛又爽又黄的视频吉 | 亚洲国产综合在线观看不卡 | 国产激情在线视频 | 日本高清中文字幕在线 | 最好看的中文字幕高清电影 | 亚洲专区欧美三级 | 91大片淫黄大片.在线天堂 | 少女哔哩| 欧美一级a一级a爱片免费免免 | 国产女主播午夜福利在线观看 | 亚洲国产精品欧美日韩一区二区 | 曰本性l交片视频免费 | 国产高清在线丝袜精品一区 | 国产欧美精品一区二区三区pp | 亚洲无卡免费 | 国产黄大片在线观看 | 玩肥熟老妇bbwxxx视频 | 日本欧美三级r级国产在线 亚洲激情乱伦 | 国产系列丝袜熟女精品网站 | 人成精品视频三区二区一区 | 国产精品多p对白交换绿帽 国产日本韩国视频 | 91极品美 | 国产精品亚洲给色区 | 在线观看福利影院 | 国产亚洲欧美另类一区 | 老司机精品| 国产鲁鲁视频在线观看 | 国产亚洲欧美丝袜 | 欧美日韩在线在线观看 | 每日更新在线观看 | 成电影在线观看 | 国产精品亚洲а∨天堂网不卡 | 欧美日韩精品一区二区在线播放蜜 | 美女是黄的网址视频在线 | 色偷偷中文字 | 精品国产aⅴ一区二区三区四川人 | 91丝袜在线观看亚洲 | 91欧美在线视频 | 香蕉香蕉国产片一级一级毛 | 国产一级高 | 欧美一区在线播放 | 超级媚药痉 | 一级特黄性色生活片一区二区 | 免费99精品国产 | 日韩欧美三区免费观看 | 日本免费精品 | 国产高清亚洲精品视bt天堂频 | 琪琪午夜福利免费院 | 丰满的女房东在线观看6 | 欧美精品一二三区 | 国产精品成人免费视频99 | 国精品一区二区三区免费观看 | 久青草国产在视频青草99在 | 国产va精品免费在线观看 | 国产午夜福利精品一 | 真实国产熟睡乱子伦视频 | 强伦轩一区二区三区四区播放方式 | 日韩亚洲产在线观看 | 国产精品亚洲一区二区三区在线 | 白丝袜国产淫秽网站 | 亚洲熟女乱色一区二区三区 | 亚洲电影在线免费观看 | 日韩免费在线 | 日韩精品一区二区三区影院 | 日韩一区二区三区免费网站 | 银杏视频在线官网 | 99热永久地址有精品 | 国产亚洲一区在线 | 99热在线 | 亚洲熟女乱综合一区 | 97精品国产高清自在线看超 | 大陆国语自产精品视频在 | 精品国内一区二区三区免费 | 香蕉一区二区在线观看 | 伦理、限制级电影手机在线观看 | 97精品视频在线观看 | 最新国产网红 | 正在播放国产多p交换视频 日韩成全视频观看免费观看高清 | 亚洲精品亚洲精品亚洲精品日韩 | 国产欧美一区二区高清在线 | 国产一级特黄aaa大片在线观 | 中文字幕在线观看亚洲 | 天下第一日本在线观看视频 | 国产精品亚洲社区在线观看 | 精品免费美剧网排行榜在线看 | 日本免费一区视频 | 国产精品亚洲专区一区 | 午夜免费福 | 亚洲国内自拍愉拍中文字幕 | 在线视频欧美日韩 | 男人性毛 | 丁香花在线视频观看免费 | 在线欧美精品二区三区 | 国产亚洲精品国产福利在线观看 | 亚洲欧美日韩激情在线观 | 精品不卡一区二区 | 午夜国产福利在线直播 | 最新国产ts人妖系列视频 | 国产91精品系 | 丰乳翘臀 | 涩涩欧美 | 亚洲品质自拍视频 | 热门好看动漫综艺 | 18出禁止看的啪视频网站 | 青青草97国产精品免费观看 | 欧美精品专区在线视频 | 国产三j | 国产精品一区二区 | 在线观看国产一级 | 日本三级强在线观看 | 亚洲综合在线一区二区三区 | 免费在线观看a视频 | 欧美性夜影院亚洲 | 免费看老女人 | 国产91小视频在线观看 | 日韩免费一区二区三区高清 | 国产新拍在线 | 国产老妇伦国产熟女中文视频 | 在线人成观看 | 国产亚洲精在线看 | 精品国产乱 | 不卡中文字幕激情视频网站 | 精品深夜| 国产高清在线精品一区免费97 | 国产精品一区高清在线观看 | 国产在线播精品第三 | 亚洲精品熟女中文字幕 | 中文字幕日产熟女乱码 | 亚洲精品一品区二品区三品区 | 国产精品综合色区在线 | 国产精品第1页 | 日本三级a∨在线观看 | 亚洲精品日韩一区 | 玩肥熟老妇bbwxxx视频 | 91日韩高清在线观看播放 | 小草国产精品情侣 | 欧美a级片视频 | 国产精品6| 潦草影院 | 含羞草国产亚洲精品岁国产精品 | 国产狂喷潮在线观看中文 | 国产自经典三级在线观看 | 国产一级特黄高清在线大片 | 国产男同gaya | 国产精品爽爽va在 | 精品视频国产激情 | 热99re6久精品国产首页青柠 | 日韩精品国产一区 | 成年人24小时在线免费观看视频 | 日本三级香港三级人妇99 | 99在线精品日韩一区免费国产 | 午夜福利电影 | 精品国产免费人成电影在线观 | 欧美日韩在线播一区二区三区 | 免费人成在线视频无 | 三区在线观看不卡 | 热门电视剧 | 欧美怡红院免费全部视频 | 香港午夜三级a三级高清观看 | 中文亚洲欧美日韩无线码 | 国产人在线成 | 成人短视频黄 | 亚洲无线观看国产高 | 在线亚洲欧美日韩每日更新 | 国产午夜福利100集发布 | 观看视频新选择 | 正在播放国产精品 | 日本a优不卡在线播放 | 国产一区二区在免费观看 | 成人国产一区二区三区 | 国产偷国产偷亚洲清高app | 午夜视频在线观看国产 | 国产十八 | 欧洲亚洲欧美国产日本高清 | 免费看美女脱了全身衣服直播 | 福利一区二区三区视频在线观看 | 老司机精 | 香蕉在线一本大道 | 亚洲成a人片 | 一级a一片在线播放国产 | 亚洲日本乱伦中文 | 日韩精品一区二区三区视频网 | 国产精品亚洲无线码在线播放 | 中文字幕九热精品视频在线 | 欧美人与性动交α欧美精品 | 日本日本乱码伦视频在线 | 亚洲制服国产一区二区三区 | 果冻传媒视频一二在线观看 | 亚洲人成影视在线观看 | 国产精品亚洲自在线播放页码 | 精品热亚洲一级 | 亚洲国产人成自精在线尤物 | 99精品欧美一区二 | 日本三级在线 | 日本高清色www在线安全 | 老女肥女熟国产在线视频网址 | 在线观看有 | 精品国产yw在线观看 | 国产尤物在线视精品亚洲 | 亚洲欧美成va人在线观看 | 亚洲vr| 欧美午夜理伦三级在线观看 | 国产一级淫片视频免费看 | 国产欧美日韩精品视频二区 | 国产主播一区二区三区在线观 | 亚洲国产中日韩精品综合 | 国产36页在线 | 曰韩精品 | 区二区视频免费看 | 欧洲精品不卡1卡2卡三卡 | 午夜伦情电午夜伦情电影 | 尤物免费人成在线观看播放a | 微博网红户外露出在线观看 | 看看永久 | 国产乱子伦精品视频 | 日韩不卡手机视频在线观看 | 私人影院 | 中文字字幕乱码无线精品精品 | 免费人成年短视频免费网站 | 天天澡日日澡狠狠欧美老妇 | 欧美一级二级三级在线看 | 日本va在线视频播放 | 欧美又大 | 人人综合亚洲 | 国产欧美国产精品第一区 | 免费二级c片在线观看a | 国产高清一区二区在线免费观看 | 园内精品自拍视频在线播放 | 亚洲制服在线日韩 | 老司机91精品网站在线观看 | 国产福利一区二区三区四区 | 亚洲中文字幕乱码一区 | 中文字幕制 | 亚洲国产精品综 | 中文字幕永久一区二区三区 | 五月丁香婷婷综合激情在线 | 一品二品国精破解 | 免费三级在线观看中文字幕 | 青青久热 | 中文字幕不卡高 | 午夜福利电影在线观看 | 亚洲欧美中文字幕在线一区 | 超前点播最新电影电视剧 | 亚洲国产欧美日韩精品18 | 国产一区欧美亚洲 | 影音先锋在 | 在线观看高清无 | 亚洲va天堂va欧美ⅴa | 欧洲尺码日本尺码专线 | 国产综合视频在线观看8 | 亚洲人成欧美中文字幕 | 日本三级理论一区二区三区 | 国内精品国语自产拍在线观看91 | 日本护士视频欧美无砖专区 | 亚洲欧美另类专区 | 中文韩国午夜理伦三级好看 | 日本成人频道一区二区三区 | 国产va免费精品 | 国产精品xxxx国产喷水 | 一区二区三区高清视频国产女人 | 亚洲中文字幕无线 | 国产在线精品成人一区二区三区 | 国产男生午夜福利免费网站 | 最新国产一区二 | 国内自拍第一页 | 婷婷综合缴情亚洲狠狠 | 国产深夜在线免费观看 | 国产免费一区二区三区免费视频 | 国产精品女主播主要上线 | 激情小说免费阅读 | 欧美日韩国产一区二区三区不卡 | 激情欧美经典日韩 | 国产一区二区免费播放 | 国产精品亚洲视频在线观看 | 日本在线视频高清不卡 | 国产亚洲精品午夜高清影院 | 三级影视 | 视频一区视频二区在线观看 | 亚洲欧美日韩国产精品专区网 | 最新欧美日韩 | 国产女技师按摩在线观看 | 国产91精品露脸国语对白 | 96福利国产在线 | 国产最新一区二区三区天堂 | 亚洲国产欧美在线人成aaaa | 久精品视在线观看视频 | 星辰影视大全免费版官网 | 日日噜噜夜夜狠狠视频无 | 精品三级影视在线免费观看 | 亚洲精品国产自在在线观看 | 免费人成网站在线观看欧美 | 国产福利小视频在线免费观看 | 国产频99热精品在线 | 亚洲第一香蕉 | 不卡一卡 | 国产熟女熟女 | 女人一区二区视频免费 | 国产吹潮视频在线观看 | 国产精品国色 | 噜噜噜在线视频免费观看 | 亚洲日韩电影网天堂影院 | 91啪国自产最新91啪国自产 | anquye| 一区二区三区影院在线午夜 | 亚洲中国久 | 91精品国产自产91精品 | 国产日产欧美一 | 国产高清成免费视频 | 97国产在线视频 | 日韩精品国产一级 | 亚洲国产高清在线不卡 | 欧美人另是日本人妖 | 中文有码 | 4480yy午夜私人影院 | 国产免费观看青青草原网站 | 韩国中文全部三级伦在线观看中文 | 色老板在线永免费观看 | 在线精品亚洲一 | 730099带你另眼看世界 | 欧美亚洲日本中文字幕在线 | 在线观看精品国产福利片87 | 亚洲国产综合人成综合网站 | 午夜福利精品在线播放 | 欧美曰韩免费一级在线 | 国产亚洲中文不卡二区 | 久热爱精 | 国产高清在线精品二区一 | 国产在线精品一区二区夜色 | 国产精品日韩精品 | 热播电视剧免费在线观看 | 国产片网站 | 欧美制服丝袜国产日韩一区 | 成人免费观看男女 | 亚洲精品自拍视频在线免费 | 日韩视频免费在线观看 | 天下第一社区在线观看视频 | 欧美日韩一区二区精品 | 亚洲国产一区二区三区亚瑟 | 国产亚洲一区二区在线 | 精品视频二区 | 国产在线欧美日韩精品一区 | 国产精品福利尤物youwu | 午夜福利| 国产乱人视频免费播放 | 国产亚洲一区激情小说 | 国产精品v欧美精品v日韩 | 级日本乱| 欧美精品成人3d在线 | 日本一区二区三区在线观看入口 | 国产高清尿小便嘘嘘视频 | 中文字幕2025 | 免费人成在线观看播放 | 亚洲一区二区在线免费观看 | 国产精品手机免费 | 日韩精品一区二区三区免费视频 | 国产日韩精品一区二区三区在线观 | 国产亚洲综合区成 | 国内激情自拍 | 国产高清吹潮免费视频 | 国产伦精| 国产日韩一区美利坚 | 国产综合亚洲欧美日韩一区二区 | 精品三级视频在线观看 | 日本一本免费线观看视频 | 午夜福利| 日本成本人片视频免费 | 欧美性白人极 | 夜色国产精品欧美在线观看 | 国产一在线精品一区在线观看 | 国产激情国语对白 | 中文乱码字幕在线观看播放 | 小草青青手机免费视频影院 | 亚洲精品欧美精品日韩精品 | 国产精品高清在线看 | 亚洲精品午夜福利片 | 亚洲精品熟女 | 国产免费a视频 | 午夜国产福利在线 | 樱花动漫网站官网 | 欧美一级欧美一级高清 | 国产黑色丝袜在线看片不卡顿 | 99久热 | 一二三区在线播放国内精品自产拍 | 亚洲精品天堂在 | 欧美日本国 | 亚洲高清免费观看 | 国产91精品在线 | 欧美丝袜国 | 国产亚洲精品资源在 | 精品综合国产一区二区三区码码 | 国农村精品国产自线拍 | 全集高清免费的影视剧在线观看 | 精品无人区一区二区三区 | 91国内精品在线 | 国产91丝袜在线播放0 | 欧美精品一区二区三区视频 | 欧美日韩国产一区二区三区在 | 国产精品偷伦 | 色国产综合免费视频在线播放 | 国产性爱在线观看 | 国产免费观看青青草原网站 | 免费看黄的网址 | 羞羞视频下载 | 国产在线观看精品一区二区三区 | 99一区二区三区国产热视频在线 | 国产人在线成 | 欧美日韩国产色综合一二三四 | 真实国产普通话对白乱子子伦视频 | 中文乱伦综合高清 | 最近日本韩国观看 | 亚洲国产欧美在线一区二区 | 美国十次狠狠 | 国产丝袜视频在线 | 国产在线观看中文字幕 | 污污污污污免费网站在线观看 | 国产丝袜精品 | 国产最新一区二区三区天堂 | 91牛牛国产在线无弹窗 | 电视剧大全免 | 亚洲国产日韩无在线播放 | 91日本在线精品高清观看 | 亚洲日本欧美日韩在线观看 | 另类尿喷潮videofree | 中文乱伦综合高清 | 亚洲一区二区三区中文字幕在线 | 一区二区免费高清在线观看国产 | 最近最好的2025中文日本字幕 | 欧美丰满老妇熟乱xxxxx视频 | 永久免费a | 一区两区三区 | 国产片侵 | 欧美日韩免费一区二区在线观看 | 午夜影院0606免费 | 国产色系视频免费在线观看 | yellow高清免费观看日本 | 国产亚洲欧美在线播放网站 | 动漫无遮 | 国内精品自在自线视频香蕉 | 最新好看的影视大全在线观看 | 国产日韩精品一区二区在线观看 | 欧美日韩在线亚洲国产人 | 欧美网址在线观看 | 老太xxxx下面毛茸茸 | 精品欧美一区二区三区在线观看 | 精品国产日韩一区三区 | 亚洲一区二 | 山东猎头 | 国产亚洲精品综合网在线观看 | 手机看片日韩1024 | 成人午夜一区二区三区视频 | 999zyz玖玖资源站永久 | 国产午夜福利不卡在线观看 | 在线在线播放 | 亚洲国产v高清在线观看 | 国产精品福利短视在线播放频 | 国产乱子伦露脸在线 | 欧美国产日韩一区二区三区综合视 | 国产精品一区视频 | 国产麻传媒精品国产v | 咻咻漫画在线观看 | 2025年精品国产福利在线 | 国产精彩对白在线91 | 女人与动zz |