Swift新特性 — 訪問控制(Access Control)

蘋果在發(fā)布了Xcode 6 Bate 4后為Swift添加了新的特性–訪問控制(Access Control),并且更新了The Swift Programming Language文檔,我抽空把這篇文檔翻譯了一下,下面讓我們來詳細(xì)了解一下Access Control。

訪問控制

訪問控制可以限定你在源文件或模塊中訪問代碼的級別,也就是說可以控制哪些代碼你可以訪問,哪些代碼你不能訪問。這個特性可以讓我們隱藏功能實(shí)現(xiàn)的一些細(xì)節(jié),并且可以明確的指定我們提供給其他人的接口中哪些部分是他們可以使用的,哪些是他們看不到的。

你可以明確的給類、結(jié)構(gòu)體、枚舉、設(shè)置訪問級別,也可以給屬性、函數(shù)、初始化方法、基本類型、下標(biāo)索引等設(shè)置訪問級別。協(xié)議也可以被限定在一定的范圍內(nèi)使用,包括協(xié)議里的全局常量、變量和函數(shù)。

在提供了不同訪問級別的同時,Swift并沒有規(guī)定我們要在任何時候都要在代碼中明確指定訪問級別。其實(shí),如果我們作為獨(dú)立開發(fā)者在開發(fā)我們自己的app,而不是在開發(fā)一些Framework的時候,我們完全可以不用明確的指定代碼的訪問級別。

注意:為方便起見,在代碼中可以設(shè)置訪問級別的它們(屬性、基本類型、函數(shù)等)在下面的章節(jié)中我們稱之為“實(shí)體”。

模塊和源文件

Swift中的訪問控制模型基于模塊和源文件這兩個概念。

模塊指的是Framework或App bundle。在Swift中,可以用import關(guān)鍵字引入自己的工程。在Swift中,F(xiàn)ramewordk或App bundle被作為模塊處理。如果你是為了實(shí)現(xiàn)某個通用的功能,或者是為了封裝一些常用方法而將代碼打包成Framework,這個Framework在Swift中就被稱為模塊。不論它被引入到某個App工程或者其他的Framework,它里面的一切(屬性、函數(shù)等)都屬于這個模塊。

源文件指的是Swift中的Swift File,就是編寫Swift代碼的文件,它通常屬于一個模塊。通常一個源文件包含一個類,在類中又包含函數(shù)、屬性等類型。

訪問級別

Swift提供了三種不同的訪問級別。這些訪問級別相對于源文件中定義的實(shí)體,同時也相對于這些源文件所屬的模塊。

Public:可以訪問自己模塊或應(yīng)用中源文件里的任何實(shí)體,別人也可以訪問引入該模塊中源文件里的所有實(shí)體。通常情況下,某個接口或Framework是可以被任何人使用時,你可以將其設(shè)置為public級別。

Internal:可以訪問自己模塊或應(yīng)用中源文件里的任何實(shí)體,但是別人不能訪問該模塊中源文件里的實(shí)體。通常情況下,某個接口或Framework作為內(nèi)部結(jié)構(gòu)使用時,你可以將其設(shè)置為internal級別。

Private:只能在當(dāng)前源文件中使用的實(shí)體,稱為私有實(shí)體。使用private級別,可以用作隱藏某些功能的實(shí)現(xiàn)細(xì)節(jié)。

Public為最高級訪問級別,Private為最低級訪問級別。

訪問級別的使用原則

在Swift中,訪問級別有如下使用原則:訪問級別統(tǒng)一性。 比如說:

一個public訪問級別的變量,不能將它的類型定義為internal和private的類型。因?yàn)樽兞靠梢员蝗魏稳嗽L問,但是定義它的類型不可以,所以這樣就會出現(xiàn)錯誤。
函數(shù)的訪問級別不能高于它的參數(shù)、返回類型的訪問級別。因?yàn)槿绻瘮?shù)定義為public而參數(shù)或者返回類型定義為internal或private,就會出現(xiàn)函數(shù)可以被任何人訪問,但是它的參數(shù)和返回類型不可以,同樣會出現(xiàn)錯誤。

默認(rèn)訪問級別

代碼中的所有實(shí)體,如果你不明確的定義其訪問級別,那么它們默認(rèn)為internal級別。在大多數(shù)情況下,我們不需要明確的設(shè)置實(shí)體的訪問級別,因?yàn)槲覀兇蠖鄶?shù)時候都是在開發(fā)一個App bundle。

單目標(biāo)應(yīng)用程序的訪問級別

當(dāng)你編寫一個單目標(biāo)應(yīng)用程序時,該應(yīng)用的所有功能都是為該應(yīng)用服務(wù),不需要提供給其他應(yīng)用或者模塊使用,所以我們不需要明確設(shè)置訪問級別,使用默認(rèn)的訪問級別internal即可。但是如果你愿意,你也可以使用private級別,用于隱藏一些功能的實(shí)現(xiàn)細(xì)節(jié)。

Framework的訪問級別

當(dāng)你開發(fā)Framework時,就需要把一些實(shí)體定義為public級別,以便其他人導(dǎo)入該Framework后可以正常使用其功能。這些被你定義為public的實(shí)體,就是這個Framework的API

注意:Framework的內(nèi)部實(shí)現(xiàn)細(xì)節(jié)依然可以使用默認(rèn)的internal級別,或者也可以定義為private級別。只有你想將它作為API的實(shí)體,才將其定義為public級別。

訪問控制語法

通過修飾符public、internal、private來聲明實(shí)體的訪問級別:

public class SomePublicClass {}
internal class SomeInternalClass {}
private class SomePrivateClass {}

public var somePublicVariable = 0
internal let someInternalConstant = 0
private func somePrivateFunction() {}

除非有特殊的說明,否則實(shí)體都使用默認(rèn)的訪問級別internal,可以查閱默認(rèn)訪問級別這一節(jié)。這意味著SomeInternalClass和someInternalConstant不用明確的使用修飾符聲明訪問級別,但是他們?nèi)稳粨碛须[式的訪問級別internal:

class SomeInternalClass {}              // 隱式訪問級別internal
var someInternalConstant = 0            // 隱式訪問級別 internal
自定義類型

如果你想為一個自定義類型指定一個明確的訪問級別,那么你要明確一點(diǎn)。那就是你要確保新類型的訪問級別和它實(shí)際的作用域相匹配。比如說,如果某個類里的屬性、函數(shù)、返回值它們的作用域僅在當(dāng)前的源文件中,那么你就可以將這個類申明為private類,而不需要申明為public或者internal類。

類的訪問級別也可以影響到類成員(屬性、函數(shù)、初始化方法等)的默認(rèn)訪問級別。如果你將類申明為private類,那么該類的所有成員的默認(rèn)訪問級別也會成為private。如果你將類申明為public或者internal類(或者不明確的指定訪問級別,而使用默認(rèn)的internal訪問級別),那么該類的所有成員的訪問級別是internal。

注意:上面提到,一個public類的所有成員的訪問級別默認(rèn)為internal級別,而不是public級別。如果你想將某個成員申明為public級別,那么你必須使用修飾符明確的申明該成員。這樣做的好處是,在你定義公共接口API的時候,可以明確的選擇哪些屬性或方法是需要公開的,哪些是內(nèi)部使用的,可以避免將內(nèi)部使用的屬性方法公開成公共API的錯誤。

public class SomePublicClass {          // 顯示的 public 類
    public var somePublicProperty = 0    // 顯示的 public 類成員
    var someInternalProperty = 0         // 隱式的 internal 類成員
    private func somePrivateMethod() {}  // 顯示的 private 類成員
}

class SomeInternalClass {               // 隱式的 internal 類
    var someInternalProperty = 0         // 隱式的 internal 類成員
    private func somePrivateMethod() {}  // 顯示的 private 類成員
}

private class SomePrivateClass {        // 顯示的 private 類
    var somePrivateProperty = 0          // 隱式的 private 類成員
    func somePrivateMethod() {}          // 隱式的 private 類成員
}
元組類型

元組的訪問級別使用是所有類型的訪問級別使用中最為嚴(yán)謹(jǐn)?shù)摹1热缯f,如果你構(gòu)建一個包含兩種不同類型元素的元組,其中一個元素類型的訪問級別為internal,另一個為private級別,那么這個元組的訪問級別為private。也就是說元組的訪問級別遵循它里面元組中最低級的訪問級別。

注意:元組不同于類、結(jié)構(gòu)體、枚舉、函數(shù)那樣有單獨(dú)的定義。元組的訪問級別是在它被使用時自動推導(dǎo)出的,而不是明確的申明。

函數(shù)類型

函數(shù)的訪問級別需要根據(jù)該函數(shù)的參數(shù)類型訪問級別、返回類型訪問級別得出。如果根據(jù)參數(shù)類型和返回類型得出的函數(shù)訪問級別不符合上下文,那么就需要明確的申明該函數(shù)的訪問級別。

下面的例子中定義了一個全局函數(shù)名為someFunction,并且沒有明確的申明其訪問級別。你也許會認(rèn)為該函數(shù)應(yīng)該擁有默認(rèn)的訪問級別internal,但事實(shí)并非如此。事實(shí)上,如果按下面這種寫法,編譯器是無法編譯通過的:

 func someFunction() -> (SomeInternalClass, SomePrivateClass) {
    // function implementation goes here
}

我們可以看到,這個函數(shù)的返回類型是一個元組,該元組中包含兩個自定義的類(可查閱自定義類型)。其中一個類的訪問級別是internal,另一個的訪問級別是private,所以根據(jù)元組訪問級別的原則,該元組的訪問級別是private(元組的訪問級別遵循它里面元組中最低級的訪問級別)。因?yàn)樵摵瘮?shù)返回類型的訪問級別private
,所以你必須使用private修飾符,明確的申請?jiān)摵瘮?shù):

private func someFunction() -> (SomeInternalClass, SomePrivateClass) {
    // function implementation goes here
}

將該函數(shù)申明為public或internal,或者使用默認(rèn)的訪問級別internal都是錯誤的,因?yàn)槿绻言摵瘮?shù)當(dāng)做public或internal級別來使用的話,是無法得到private級別的返回值的。

枚舉類型

枚舉中成員的訪問級別繼承自該枚舉,你不能為枚舉中的成員指定訪問級別。
比如下面的例子,枚舉CompassPoint被明確的申明為public級別,那么它的成員North,South,East,West的訪問級別同樣也是public:

 public enum CompassPoint {
    case North
    case South
    case East
    case West
}
原始值和關(guān)聯(lián)值

用于枚舉定義中的任何原始值,或關(guān)聯(lián)的值類型必須有一個訪問級別,至少要高于枚舉的訪問級別。比如說,你不能在一個internal訪問級別的枚舉中定義private級別的原始值類型。

嵌套類型

如果在private級別的類型中定義嵌套類型,那么該嵌套類型就自動擁有private訪問級別。如果在public或者internal級別的類型中定義嵌套類型,那么該嵌套類型自動擁有internal訪問級別。如果想讓嵌套類型擁有public訪問級別,那么需要對該嵌套類型進(jìn)行明確的訪問級別申明

子類

子類的訪問級別不得高于父類的訪問級別。比如說,父類的訪問級別是internal,子類的訪問級別就不能申明為public。

此外,在滿足子類不高于父類訪問級別以及遵循各訪問級別作用域(即模塊或源文件)的前提下,你可以重寫任意類成員(方法、屬性、初始化方法、下標(biāo)索引等)。

<u>如果我們無法直接訪問某個類中的屬性或函數(shù)等,那么可以繼承該類,從而可以更容易的訪問到該類的類成員。</u>下面的例子中,類A的訪問級別是public,它包含一個函數(shù)someMethod,訪問級別為private。類B繼承類A,并且訪問級別申明為internal,但是在類B中重寫了類A中訪問級別為private的方法someMethod,并重新申明為internal級別。通過這種方式,我們就可以訪問到某類中private級別的類成員,并且可以重新申明其訪問級別,以便其他人使用:

public class A {
    private func someMethod() {}
}

internal class B: A {
    override internal func someMethod() {}
}

只要滿足子類不高于父類訪問級別以及遵循各訪問級別作用域的前提下<u>(即private的作用域在同一個源文件中,internal的作用域在同一個模塊下),我們甚至可以在子類中,用子類成員訪問父類成員,哪怕父類成員的訪問級別比子類成員的要低:</u>

 public class A {
    private func someMethod() {}
}

internal class B: A {
    override internal func someMethod() {
        super.someMethod()
    }
}

因?yàn)楦割怉和子類B定義在同一個源文件中,所以在類B中可以在重寫的someMethod方法中調(diào)用super.someMethod()。

常量、變量、屬性、下標(biāo)

常量、變量、屬性、下標(biāo)索引的Getters和Setters的訪問級別繼承自它們所屬成員的訪問級別。
Setter的訪問級別可以低于對應(yīng)的Getter的訪問級別,這樣就可以控制變量、屬性或下標(biāo)索引的讀寫權(quán)限。在var或subscript定義作用域之前,你可以通過private(set)或internal(set)先為它門的寫權(quán)限申明一個較低的訪問級別。

注意:這個規(guī)定適用于用作存儲的屬性或用作計(jì)算的屬性。即使你不明確的申明存儲屬性的Getter、Setter,Swift也會隱式的為其創(chuàng)建Getter和Setter,用于對該屬性進(jìn)行讀取操作。使用private(set)和internal(set)可以改變Swift隱式創(chuàng)建的Setter的訪問級別。在計(jì)算屬性中也是同樣的。
TrackedString結(jié)構(gòu)體定義了一個用于存儲的屬性名為value,類型為String,并將初始化值設(shè)為""(即一個空字符串)。該結(jié)構(gòu)體同時也定義了另一個用于存儲的屬性名為numberOfEdits,類型為Int,它用于記錄屬性value被修改的次數(shù)。這個功能的實(shí)現(xiàn)通過屬性value的didSet方法實(shí)現(xiàn),每當(dāng)給value賦新值時就會調(diào)用didSet方法,給numberOfEdits加一。

結(jié)構(gòu)體TrackedString和它的屬性value均沒有明確的申明訪問級別,所以它們都擁有默認(rèn)的訪問級別internal。但是該結(jié)構(gòu)體的numberOfEdits屬性使用private(set)修飾符進(jìn)行申明,這意味著numberOfEdits屬性只能在定義該結(jié)構(gòu)體的源文件中賦值。numberOfEdits屬性的Getter依然是默認(rèn)的訪問級別internal,但是Setter的訪問級別是private,這表示該屬性只有在當(dāng)前的源文件中是可讀可寫的,在當(dāng)前源文件所屬的模塊中它只是一個可讀的屬性。

如果你實(shí)例化TrackedString結(jié)構(gòu)體,并且多次對value屬性的值進(jìn)行修改,你就會看到numberOfEdits的值會隨著修改次數(shù)更改:

var stringToEdit = TrackedString()
stringToEdit.value = "This string will be tracked."
stringToEdit.value += " This edit will increment numberOfEdits."
stringToEdit.value += " So will this one."
println("The number of edits is \(stringToEdit.numberOfEdits)")
// prints "The number of edits is 3"

雖然你可以在其他的源文件中實(shí)例化該結(jié)構(gòu)體并且獲取到numberOfEdits屬性的值,但是你不能對其進(jìn)行賦值。這樣就能很好的告訴使用者,你只管使用,而不需要知道其實(shí)現(xiàn)細(xì)節(jié)。

初始化

我們可以給自定義的初始化方法指定訪問級別,但是必須要低于或等于它所屬類的訪問級別。但如果該初始化方法是必須要使用的話,那它的訪問級別就必須和所屬類的訪問級別相同。

如同函數(shù)或方法參數(shù),初始化方法參數(shù)的訪問級別也不能低于初始化方法的訪問級別。

默認(rèn)初始化方法

Swift為結(jié)構(gòu)體、類都提供了一個默認(rèn)的無參初始化方法,用于給它們的所有屬性提供賦值操作,但不會給出具體值。默認(rèn)初始化方法可以參閱Default Initializers。默認(rèn)初始化方法的訪問級別與所屬類型的訪問級別相同。

注意:如果一個類型被申明為public級別,那么默認(rèn)的初始化方法的訪問級別為internal。如果你想讓無參的初始化方法在其他模塊中可以被使用,那么你必須提供一個具有public訪問級別的無參初始化方法。

結(jié)構(gòu)體的默認(rèn)成員初始化方法

如果結(jié)構(gòu)體中的任一存儲屬性的訪問級別為private,那么它的默認(rèn)成員初始化方法訪問級別就是private。盡管如此,結(jié)構(gòu)體的初始化方法的訪問級別依然是internal。

如果你想在其他模塊中使用該結(jié)構(gòu)體的默認(rèn)成員初始化方法,那么你需要提供一個訪問級別為public的默認(rèn)成員初始化方法。

協(xié)議

如果你想為一個協(xié)議明確的申明訪問級別,那么有一點(diǎn)需要注意,就是你要確保該協(xié)議只在你申明的訪問級別作用域中使用。
協(xié)議中的每一個必須要實(shí)現(xiàn)的函數(shù)都具有和該協(xié)議相同的訪問級別。這樣才能確保該協(xié)議的使用者可以實(shí)現(xiàn)它所提供的函數(shù)。

注意:如果你定義了一個public訪問級別的協(xié)議,那么實(shí)現(xiàn)該協(xié)議提供的必要函數(shù)也會是public的訪問級別。這一點(diǎn)不同于其他類型,比如,public訪問級別的其他類型,他們成員的訪問級別為internal。

協(xié)議繼承

如果定義了一個新的協(xié)議,并且該協(xié)議繼承了一個已知的協(xié)議,那么新協(xié)議擁有的訪問級別最高也只和被繼承協(xié)議的訪問級別相同。比如說,你不能定義一個public的協(xié)議而去繼承一個internal的協(xié)議。

協(xié)議一致性

類可以采用比自身訪問級別低的協(xié)議。比如說,你可以定義一個public級別的類,可以讓它在其他模塊中使用,同時它也可以采用一個internal級別的協(xié)議,并且只能在定義了該協(xié)議的模塊中使用。

采用了協(xié)議的類的訪問級別遵循它本身和采用協(xié)議中最低的訪問級別。也就是說如果一個類是public級別,采用的協(xié)議是internal級別,那個采用了這個協(xié)議后,該類的訪問級別也是internal。

如果你采用了協(xié)議,那么實(shí)現(xiàn)了協(xié)議必須的方法后,該方法的訪問級別遵循協(xié)議的訪問級別。比如說,一個public級別的類,采用了internal級別的協(xié)議,那么該類實(shí)現(xiàn)協(xié)議的方法至少也得是internal。

注意:在Swift中和Objective-C中一樣,協(xié)議的一致性保證了一個類不可能在同一個程序中用不同的方法采用同一個協(xié)議。

擴(kuò)展

你可以在條件允許的情況下對類、結(jié)構(gòu)體、枚舉進(jìn)行擴(kuò)展。擴(kuò)展成員應(yīng)該具有和原始類成員一致的訪問級別。比如你擴(kuò)展了一個公共類型,那么你新加的成員應(yīng)該具有和原始成員一樣的默認(rèn)的internal訪問級別。

或者,你可以明確申明擴(kuò)展的訪問級別(比如使用private extension)給該擴(kuò)展內(nèi)所有成員指定一個新的默認(rèn)訪問級別。這個新的默認(rèn)訪問級別仍然可以被單獨(dú)成員所指定的訪問級別所覆蓋。

協(xié)議的擴(kuò)展

如果一個擴(kuò)展采用了某個協(xié)議,那么你就不能對該擴(kuò)展使用訪問級別修飾符來申明了。該擴(kuò)展中實(shí)現(xiàn)協(xié)議的方法都會遵循該協(xié)議的訪問級別。

泛型

泛型類型或泛型函數(shù)的訪問級別遵循泛型類型、函數(shù)本身、泛型類型參數(shù)三者中訪問級別最低的級別。

類型別名

任何被你定義的類型別名都會認(rèn)為是不同的類型進(jìn)行訪問控制。一個類型別名的訪問級別低于或等于這個類型的訪問級別。比如說,一個private級別的類型別名可以設(shè)定給一個public、internal、private的類型,但是一個public級別的類型別名只能設(shè)定給一個public級別的類型,不能設(shè)定給internal或private的類類型。

注意:這條規(guī)則也適用于為滿足協(xié)議一致性而給相關(guān)類型命名別名。

相關(guān)鏈接
The Swift Programming Language
Using Swift with Cocoa and Objective-C

本文由宇軒翻譯自Apple Swift Blog,英文原文地址:Access Control,譯文原文地址:Swift新特性 — 訪問控制(Access Control)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,702評論 6 534
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,615評論 3 419
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,606評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,044評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,826評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,227評論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,307評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,447評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,992評論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,807評論 3 355
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,001評論 1 370
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,550評論 5 361
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,243評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,667評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,930評論 1 287
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,709評論 3 393
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,996評論 2 374

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