Swift-循環(huán)強(qiáng)引用

最近項(xiàng)目使用的是OC咬清,后頭看之前用Swift開(kāi)發(fā)的一個(gè)項(xiàng)目時(shí)闭专,發(fā)現(xiàn)很多細(xì)節(jié)都忘記了????奴潘。
為了回憶和以后方便查看,現(xiàn)在根據(jù)官方文檔swift編程語(yǔ)言影钉,做下筆記画髓。

1、自動(dòng)引用計(jì)數(shù)

當(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)的存儲(chǔ)型屬性的值肆汹。

此外,當(dāng)實(shí)例不再被使用時(shí)予权,ARC 釋放實(shí)例所占用的內(nèi)存昂勉,并讓釋放的內(nèi)存能挪作他用。這確保了不再被使用的實(shí)例扫腺,不會(huì)一直占用內(nèi)存空間岗照。

無(wú)論你將實(shí)例賦值給屬性、常量或變量笆环,它們都會(huì)創(chuàng)建此實(shí)例的強(qiáng)引用攒至。之所以稱(chēng)之為“強(qiáng)”引用,是因?yàn)樗鼤?huì)將實(shí)例牢牢地保持住躁劣,只要強(qiáng)引用還在迫吐,實(shí)例是不允許被銷(xiāo)毀的。

class Person {
    let name: String
    init(name: String) {
        self.name = name
        print("\(name) is being initialized")
    }
    deinit {
        print("\(name) is being deinitialized")
    }
}

接下來(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")
// 打印 "John Appleseed is being initialized"

由于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)毀。

如果你將同一個(gè)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)毀

reference1 = nil
reference2 = nil

第三個(gè)也就是最后一個(gè)強(qiáng)引用被斷開(kāi)時(shí)行拢,ARC 會(huì)銷(xiāo)毀它

reference3 = nil
// 打印 "John Appleseed is being deinitialized"

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

我們可能會(huì)寫(xiě)出一個(gè)類(lèi)實(shí)例的強(qiáng)引用數(shù)永遠(yuǎn)不能變成0的代碼。如果兩個(gè)類(lèi)實(shí)例互相持有對(duì)方的強(qiáng)引用,因而每個(gè)實(shí)例都讓對(duì)方一直存在竭缝,就是這種情況房维。這就是所謂的循環(huán)強(qiáng)引用。

你可以通過(guò)定義類(lèi)之間的關(guān)系為弱引用或無(wú)主引用抬纸,以替代強(qiáng)引用咙俩,從而解決循環(huán)強(qiáng)引用的問(wèn)題。

下面展示了一個(gè)不經(jīng)意產(chǎn)生循環(huán)強(qiáng)引用的例子湿故。

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

class Apartment {
    let unit: String
    init(unit: String) { self.unit = unit }
    var tenant: Person?
    deinit { print("Apartment \(unit) is being deinitialized") }
}

var john: Person?
var unit4A: Apartment?

john = Person(name: "John Appleseed")
unit4A = Apartment(unit: "4A")
john!.apartment = unit4A
unit4A!.tenant = john

這兩個(gè)實(shí)例關(guān)聯(lián)后會(huì)產(chǎn)生一個(gè)循環(huán)強(qiáng)引用阿趁。Person實(shí)例現(xiàn)在有了一個(gè)指向Apartment實(shí)例的強(qiáng)引用,而Apartment實(shí)例也有了一個(gè)指向Person實(shí)例的強(qiáng)引用坛猪。因此脖阵,當(dāng)你斷開(kāi)john和unit4A變量所持有的強(qiáng)引用時(shí),引用計(jì)數(shù)并不會(huì)降為0墅茉,實(shí)例也不會(huì)被 ARC 銷(xiāo)毀:

john = nil
unit4A = nil

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

3、解決實(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)引用。

  • 弱引用

弱引用不會(huì)對(duì)其引用的實(shí)例保持強(qiáng)引用槐秧,因而不會(huì)阻止 ARC 銷(xiāo)毀被引用的實(shí)例啄踊。這個(gè)特性阻止了引用變?yōu)檠h(huán)強(qiáng)引用。聲明屬性或者變量時(shí)刁标,在前面加上weak關(guān)鍵字表明這是一個(gè)弱引用颠通。

因?yàn)槿跻貌粫?huì)保持所引用的實(shí)例,即使引用存在膀懈,實(shí)例也有可能被銷(xiāo)毀顿锰。因此,ARC 會(huì)在引用的實(shí)例被銷(xiāo)毀后自動(dòng)將其賦值為nil启搂。并且因?yàn)槿跻每梢栽试S它們的值在運(yùn)行時(shí)被賦值為nil硼控,所以它們會(huì)被定義為可選類(lèi)型變量,而不是常量胳赌。

當(dāng) ARC 設(shè)置弱引用為nil時(shí)牢撼,屬性觀察不會(huì)被觸發(fā)绰上。

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

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

class Apartment {
    let unit: String
    init(unit: String) { self.unit = unit }
    weak var tenant: Person?
    deinit { print("Apartment \(unit) is being deinitialized") }
}

var john: Person?
var unit4A: Apartment?

john = Person(name: "John Appleseed")
unit4A = Apartment(unit: "4A")

john!.apartment = unit4A
unit4A!.tenant = john
  • 無(wú)主引用

和弱引用類(lèi)似课锌,無(wú)主引用不會(huì)牢牢保持住引用的實(shí)例赏淌。和弱引用不同的是筛严,無(wú)主引用在其他實(shí)例有相同或者更長(zhǎng)的生命周期時(shí)使用腮考。你可以在聲明屬性或者變量時(shí),在前面加上關(guān)鍵字unowned表示這是一個(gè)無(wú)主引用曲横。

無(wú)主引用通常都被期望擁有值, 絕對(duì)不會(huì)變?yōu)閚il喂柒。不過(guò) ARC 無(wú)法在實(shí)例被銷(xiāo)毀后將無(wú)主引用設(shè)為nil,因?yàn)榉强蛇x類(lèi)型的變量不允許被賦值為nil禾嫉。

下面的例子定義了兩個(gè)類(lèi)灾杰,Customer和CreditCard,模擬了銀行客戶(hù)和客戶(hù)的信用卡熙参。這兩個(gè)類(lèi)中吭露,每一個(gè)都將另外一個(gè)類(lèi)的實(shí)例作為自身的屬性。這種關(guān)系可能會(huì)造成循環(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 { print("\(name) is being deinitialized") }
}

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

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!)

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)引用了:

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

john = nil
// 打印 "John Appleseed is being deinitialized"
// 打印 "Card #1234567890123456 is being deinitialized"
  • 無(wú)主引用以及隱式解析可選屬性

上面弱引用和無(wú)主引用的例子涵蓋了兩種常用的需要打破循環(huán)強(qiáng)引用的場(chǎng)景白筹。
1智末、Person和Apartment的例子展示了兩個(gè)屬性的值都允許為nil,并會(huì)潛在的產(chǎn)生循環(huán)強(qiáng)引用遍蟋。這種場(chǎng)景最適合用弱引用來(lái)解決吹害。
2、Customer和CreditCard的例子展示了一個(gè)屬性的值允許為nil虚青,而另一個(gè)屬性的值不允許為nil它呀,這也可能會(huì)產(chǎn)生循環(huán)強(qiáng)引用。這種場(chǎng)景最適合通過(guò)無(wú)主引用來(lái)解決棒厘。

存在著第三種場(chǎng)景纵穿,兩個(gè)屬性都必須有值,并且初始化完成后永遠(yuǎn)不會(huì)為nil奢人。在這種場(chǎng)景中谓媒,需要一個(gè)類(lèi)使用無(wú)主屬性,而另外一個(gè)類(lèi)使用隱式解析可選屬性何乎。

下面的例子定義了兩個(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
    var 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
    }
}

Country的構(gòu)造函數(shù)調(diào)用了City的構(gòu)造函數(shù)贬堵。然而恃轩,只有Country的實(shí)例完全初始化后,Country的構(gòu)造函數(shù)才能把self傳給City的構(gòu)造函數(shù)黎做。

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

由于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")
print("\(country.name)'s capital city is called \(country.capitalCity.name)")
// 打印 "Canada's capital city is called Ottawa"

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

環(huán)強(qiáng)引用還會(huì)發(fā)生在當(dāng)你將一個(gè)閉包賦值給類(lèi)實(shí)例的某個(gè)屬性,并且這個(gè)閉包體中又使用了這個(gè)類(lèi)實(shí)例時(shí)补鼻。

5哄啄、解決閉包引起的循環(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。

  • 定義捕獲列表

捕獲列表中的每一項(xiàng)都由一對(duì)元素組成州胳,一個(gè)元素是weak或unowned關(guān)鍵字冗澈,另一個(gè)元素是類(lèi)實(shí)例的引用(例如self)或初始化過(guò)的變量(如delegate = self.delegate!)。這些項(xiàng)在方括號(hào)中用逗號(hào)分開(kāi)陋葡。

如果閉包有參數(shù)列表和返回類(lèi)型,把捕獲列表放在它們前面:

lazy var someClosure: (Int, String) -> String = {
    [unowned self, weak delegate = self.delegate!] (index: Int, stringToProcess: String) -> String in
    // 這里是閉包的函數(shù)體
}

如果閉包沒(méi)有指明參數(shù)列表或者返回類(lèi)型彻采,即它們會(huì)通過(guò)上下文推斷腐缤,那么可以把捕獲列表和關(guān)鍵字in放在閉包最開(kāi)始的地方:

lazy var someClosure: Void -> String = {
    [unowned self, weak delegate = self.delegate!] in
    // 這里是閉包的函數(shù)體
}
  • 弱引用和無(wú)主引用

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

相反的岭粤,在被捕獲的引用可能會(huì)變?yōu)閚il時(shí),將閉包內(nèi)的捕獲定義為弱引用特笋。弱引用總是可選類(lèi)型剃浇,并且當(dāng)引用的實(shí)例被銷(xiāo)毀后,弱引用的值會(huì)自動(dòng)置為nil猎物。這使我們可以在閉包體內(nèi)檢查它們是否存在虎囚。

如果被捕獲的引用絕對(duì)不會(huì)變?yōu)閚il,應(yīng)該用無(wú)主引用蔫磨,而不是弱引用淘讥。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市堤如,隨后出現(xiàn)的幾起案子蒲列,更是在濱河造成了極大的恐慌窒朋,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,036評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件蝗岖,死亡現(xiàn)場(chǎng)離奇詭異侥猩,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)抵赢,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)欺劳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人瓣俯,你說(shuō)我怎么就攤上這事杰标。” “怎么了彩匕?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,411評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵腔剂,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我驼仪,道長(zhǎng)掸犬,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,622評(píng)論 1 293
  • 正文 為了忘掉前任绪爸,我火速辦了婚禮湾碎,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘奠货。我一直安慰自己介褥,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,661評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布递惋。 她就那樣靜靜地躺著柔滔,像睡著了一般。 火紅的嫁衣襯著肌膚如雪萍虽。 梳的紋絲不亂的頭發(fā)上睛廊,一...
    開(kāi)封第一講書(shū)人閱讀 51,521評(píng)論 1 304
  • 那天,我揣著相機(jī)與錄音杉编,去河邊找鬼超全。 笑死,一個(gè)胖子當(dāng)著我的面吹牛邓馒,可吹牛的內(nèi)容都是我干的嘶朱。 我是一名探鬼主播,決...
    沈念sama閱讀 40,288評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼光酣,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼见咒!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起挂疆,我...
    開(kāi)封第一講書(shū)人閱讀 39,200評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤改览,失蹤者是張志新(化名)和其女友劉穎下翎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體宝当,經(jīng)...
    沈念sama閱讀 45,644評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡视事,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,837評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了庆揩。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片俐东。...
    茶點(diǎn)故事閱讀 39,953評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖订晌,靈堂內(nèi)的尸體忽然破棺而出虏辫,到底是詐尸還是另有隱情,我是刑警寧澤锈拨,帶...
    沈念sama閱讀 35,673評(píng)論 5 346
  • 正文 年R本政府宣布砌庄,位于F島的核電站,受9級(jí)特大地震影響奕枢,放射性物質(zhì)發(fā)生泄漏娄昆。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,281評(píng)論 3 329
  • 文/蒙蒙 一缝彬、第九天 我趴在偏房一處隱蔽的房頂上張望萌焰。 院中可真熱鬧,春花似錦谷浅、人聲如沸扒俯。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,889評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)陵珍。三九已至,卻和暖如春违施,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背瑟幕。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,011評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工磕蒲, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人只盹。 一個(gè)月前我還...
    沈念sama閱讀 48,119評(píng)論 3 370
  • 正文 我出身青樓辣往,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親殖卑。 傳聞我的和親對(duì)象是個(gè)殘疾皇子站削,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,901評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容