LiveData+Retrofit網(wǎng)絡(luò)請(qǐng)求實(shí)戰(zhàn)

RxJava與Retrofit

在出現(xiàn)LiveData之前兵多,Android上實(shí)現(xiàn)網(wǎng)絡(luò)請(qǐng)求最常用的方式是使用Retrofit+Rxjava。通常是RxJavaCallAdapterFactory將請(qǐng)求轉(zhuǎn)成Observable(或者Flowable等)被觀察者對(duì)象,調(diào)用時(shí)通過(guò)subscribe方式實(shí)現(xiàn)最終的請(qǐng)求姐叁。為了實(shí)現(xiàn)線程切換日杈,需要將訂閱時(shí)的線程切換成io線程嗤栓,請(qǐng)求完成通知被觀察者時(shí)切換成ui線程。代碼通常如下:

observable.subscribeOn(Schedulers.io())
          .observeOn(AndroidSchedulers.mainThread())
          .subscribe(subscriber)

為了能夠讓請(qǐng)求監(jiān)聽(tīng)到生命周期變化治唤,onDestroy時(shí)不至于發(fā)生view空指針棒动,要需要使用RxLifecycleAutoDisposeObservable能夠監(jiān)聽(tīng)到Activity和Fragment的生命周期,在適當(dāng)?shù)纳芷谙氯∠嗛啞?/p>

LiveData與Retrofit

LiveData和Rxjava中的Observable類(lèi)似宾添,是一個(gè)被觀察者的數(shù)據(jù)持有類(lèi)船惨。但是不同的是LiveData具有生命周期感知,相當(dāng)于RxJava+RxLifecycle缕陕。LiveData使用起來(lái)相對(duì)簡(jiǎn)單輕便粱锐,所以當(dāng)它加入到項(xiàng)目中后,再使用RxJava便顯得重復(fù)臃腫了(RxJava包1~2M容量)扛邑。為了移除RxJava怜浅,我們將Retrofit的Call請(qǐng)求適配成LiveData,因此我們需要自定義CallAdapterFactory蔬崩。根據(jù)接口響應(yīng)格式不同恶座,對(duì)應(yīng)的適配器工廠會(huì)有所區(qū)別。本次便以廣為人知的wanandroid的api為例子沥阳,來(lái)完成LiveData網(wǎng)絡(luò)請(qǐng)求實(shí)戰(zhàn)跨琳。
首先根據(jù)它的響應(yīng)格式:

{
    data:[],//或者{}
    errorCode:0,
    errorMsg:""
}

定義一個(gè)通用的響應(yīng)實(shí)體ApiResponse

class ApiResponse<T>(
    var data: T?,
    var errorCode: Int,
    var errorMsg: String
)

然后我們定義對(duì)應(yīng)的LiveDataCallAdapterFactory

import androidx.lifecycle.LiveData
import retrofit2.CallAdapter
import retrofit2.Retrofit
import java.lang.reflect.Type
import retrofit2.CallAdapter.Factory
import java.lang.reflect.ParameterizedType

class LiveDataCallAdapterFactory : Factory() {
    override fun get(returnType: Type, annotations: Array<Annotation>, retrofit: Retrofit): CallAdapter<*, *>? {
        if (getRawType(returnType) != LiveData::class.java) return null
        //獲取第一個(gè)泛型類(lèi)型
        val observableType = getParameterUpperBound(0, returnType as ParameterizedType)
        val rawType = getRawType(observableType)
        if (rawType != ApiResponse::class.java) {
            throw IllegalArgumentException("type must be ApiResponse")
        }
        if (observableType !is ParameterizedType) {
            throw IllegalArgumentException("resource must be parameterized")
        }
        return LiveDataCallAdapter<Any>(observableType)
    }
}

然后在LiveDataCallAdapter將Retrofit的Call對(duì)象適配成LiveData

import androidx.lifecycle.LiveData
import retrofit2.Call
import retrofit2.CallAdapter
import retrofit2.Callback
import retrofit2.Response
import java.lang.reflect.Type
import java.util.concurrent.atomic.AtomicBoolean

class LiveDataCallAdapter<T>(private val responseType: Type) : CallAdapter<T, LiveData<T>> {
    override fun adapt(call: Call<T>): LiveData<T> {
        return object : LiveData<T>() {
            private val started = AtomicBoolean(false)
            override fun onActive() {
                super.onActive()
                if (started.compareAndSet(false, true)) {//確保執(zhí)行一次
                    call.enqueue(object : Callback<T> {
                        override fun onFailure(call: Call<T>, t: Throwable) {
                            val value = ApiResponse<T>(null, -1, t.message ?: "") as T
                            postValue(value)
                        }

                        override fun onResponse(call: Call<T>, response: Response<T>) {
                            postValue(response.body())
                        }
                    })
                }
            }
        }
    }

    override fun responseType() = responseType
}

第一個(gè)請(qǐng)求

以首頁(yè)banner接口(https://www.wanandroid.com/banner/json)為例,完成第一個(gè)請(qǐng)求桐罕。
新建一個(gè)WanApi接口脉让,加入Banner列表api,以及Retrofit初始化方法,為方便查看http請(qǐng)求和響應(yīng)功炮,加入了okhttp自帶的日志攔截器溅潜。

interface WanApi {
    companion object {
        fun get(): WanApi {
            val clientBuilder = OkHttpClient.Builder()
                .connectTimeout(60, TimeUnit.SECONDS)
            if (BuildConfig.DEBUG) {
                val loggingInterceptor = HttpLoggingInterceptor()
                loggingInterceptor.level = HttpLoggingInterceptor.Level.BODY
                clientBuilder.addInterceptor(loggingInterceptor)
            }
            return Retrofit.Builder()
                .baseUrl("https://www.wanandroid.com/")
                .client(clientBuilder.build())
                .addCallAdapterFactory(LiveDataCallAdapterFactory())
                .addConverterFactory(GsonConverterFactory.create())
                .build()
                .create(WanApi::class.java)
        }
    }
    /**
     * 首頁(yè)banner
     */
    @GET("banner/json")
    fun bannerList(): LiveData<ApiResponse<List<BannerVO>>>
}

BannerVO實(shí)體

data class BannerVO(
    var id: Int,
    var title: String,
    var desc: String,
    var type: Int,
    var url: String,
    var imagePath:String
)

我們?cè)贛ainActivity中發(fā)起請(qǐng)求

 private fun loadData() {
    val bannerList = WanApi.get().bannerList()
    bannerList.observe(this, Observer {
        Log.e("main", "res:$it")
    })
 }

調(diào)試結(jié)果如下:


banner請(qǐng)求結(jié)果

LiveData的map與switchMap操作

LiveData可以通過(guò)Transformations的map和switchMap操作,將一個(gè)LiveData轉(zhuǎn)成另一種類(lèi)型的LiveData死宣,效果與RxJava的map/switchMap操作符類(lèi)似伟恶。可以看看兩個(gè)函數(shù)的聲明

public static <X, Y> LiveData<Y> map(
            @NonNull LiveData<X> source,
            @NonNull final Function<X, Y> mapFunction)


public static <X, Y> LiveData<Y> switchMap(
            @NonNull LiveData<X> source,
            @NonNull final Function<X, LiveData<Y>> switchMapFunction)

根據(jù)以上代碼毅该,我們可以知道博秫,對(duì)應(yīng)的變換函數(shù)返回的類(lèi)型是不一樣的:map是基于泛型類(lèi)型的變換,而switchMap則返回一個(gè)新的LiveData眶掌。

還是以banner請(qǐng)求為例挡育,我們將map和switchMap應(yīng)用到實(shí)際場(chǎng)景中:
1: 為了能夠手動(dòng)控制請(qǐng)求,我們需要一個(gè)refreshTrigger觸發(fā)變量朴爬,當(dāng)這個(gè)變量被設(shè)置為true時(shí)即寒,通過(guò)switchMap生成一個(gè)新的LiveData用作請(qǐng)求banner

private val refreshTrigger = MutableLiveData<Boolean>()
private val api = WanApi.get()
private val bannerLis:LiveData<ApiResponse<List<BannerVO>>> = Transformations.switchMap(refreshTrigger) {
    //當(dāng)refreshTrigger的值被設(shè)置時(shí),bannerList
    api.bannerList()
}

2: 為了展示banner,我們通過(guò)map將ApiResponse轉(zhuǎn)換成最終關(guān)心的數(shù)據(jù)是List<BannerVO>

val banners: LiveData<List<BannerVO>> = Transformations.map(bannerList) {
    it.data ?: ArrayList()
}

LiveData與ViewModel結(jié)合

為了將LiveDataActivity解耦母赵,我們通過(guò)ViewModel來(lái)管理這些LiveData逸爵。

class HomeVM : ViewModel() {
    private val refreshTrigger = MutableLiveData<Boolean>()
    private val api = WanApi.get()
    private val bannerList: LiveData<ApiResponse<List<BannerVO>>> = Transformations.switchMap(refreshTrigger) {
        //當(dāng)refreshTrigger的值被設(shè)置時(shí),bannerList
        api.bannerList()
    }

    val banners: LiveData<List<BannerVO>> = Transformations.map(bannerList) {
        it.data ?: ArrayList()
    }

    fun loadData() {
        refreshTrigger.value = true
    }
}

在activity_main.xml中加入banner布局凹嘲,這里使用BGABanner-Android來(lái)顯示圖片

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto"
        xmlns:tools="http://schemas.android.com/tools">
    <data>
        <variable
                name="vm"
                type="io.github.iamyours.wandroid.ui.home.HomeVM"/>
    </data>
    <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:orientation="vertical">

        <cn.bingoogolapple.bgabanner.BGABanner
                android:id="@+id/banner"
                android:layout_width="match_parent"
                android:layout_height="120dp"
                android:paddingLeft="16dp"
                android:paddingRight="16dp"
                app:banner_indicatorGravity="bottom|right"
                app:banner_isNumberIndicator="true"
                app:banner_pointContainerBackground="#0000"
                app:banner_transitionEffect="zoom"/>

        <TextView
                android:layout_width="match_parent"
                android:layout_height="44dp"
                android:background="#ccc"
                android:gravity="center"
                android:onClick="@{()->vm.loadData()}"
                android:text="加載Banner"/>
    </LinearLayout>
</layout>

然后在MainActivity完成Banner初始化,通過(guò)監(jiān)聽(tīng)ViewModel中的banners實(shí)現(xiàn)輪播圖片的展示师倔。

class MainActivity : AppCompatActivity() {
    lateinit var binding: ActivityMainBinding
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = DataBindingUtil.setContentView(this, R.layout.activity_main)
        val vm = ViewModelProviders.of(this).get(HomeVM::class.java)
        binding.lifecycleOwner = this
        binding.vm = vm
        initBanner()
    }

    private fun initBanner() {
        binding.run {
            val bannerAdapter = BGABanner.Adapter<ImageView, BannerVO> { _, image, model, _ ->
                image.displayWithUrl(model?.imagePath)
            }
            banner.setAdapter(bannerAdapter)
            vm?.banners?.observe(this@MainActivity, Observer {
                banner.setData(it, null)
            })
        }
    }
}

最終效果如下:


banner

加載進(jìn)度顯示

SwipeRefreshLayout

請(qǐng)求網(wǎng)絡(luò)過(guò)程中,必不可少的是加載進(jìn)度的展示周蹭。這里我們列舉兩種常用的的加載方式趋艘,一種在布局中的進(jìn)度條(如SwipeRefreshLayout),另一種是加載對(duì)話(huà)框凶朗。
為了控制加載進(jìn)度條顯示隱藏瓷胧,我們?cè)?code>HomeVM中添加loading變量,在調(diào)用loadData時(shí)通過(guò)loading.value=true控制進(jìn)度條的顯示棚愤,在map中的轉(zhuǎn)換函數(shù)中控制進(jìn)度的隱藏

val loading = MutableLiveData<Boolean>()
val banners: LiveData<List<BannerVO>> = Transformations.map(bannerList) {
    loading.value = false
    it.data ?: ArrayList()
}
fun loadData() {
    refreshTrigger.value = true
    loading.value = true
}

我們?cè)赼ctivity_main.xml的外層嵌套一個(gè)SwipeRefreshLayout搓萧,通過(guò)databinding設(shè)置加載狀態(tài),添加刷新事件

<androidx.swiperefreshlayout.widget.SwipeRefreshLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:onRefreshListener="@{() -> vm.loadData()}"
        app:refreshing="@{vm.loading}">
        ...
</androidx.swiperefreshlayout.widget.SwipeRefreshLayout>

然后我們?cè)倏聪滦Ч?/p>

SwipeRefreshLayout進(jìn)度控制

加載對(duì)話(huà)框KProgressHUD

為了能和ViewModel解藕遇八,我們將加載對(duì)話(huà)框封裝到一個(gè)Observer中矛绘。

class LoadingObserver(context: Context) : Observer<Boolean> {
    private val dialog = KProgressHUD(context)
        .setStyle(KProgressHUD.Style.SPIN_INDETERMINATE)
        .setCancellable(false)
        .setAnimationSpeed(2)
        .setDimAmount(0.5f)

    override fun onChanged(show: Boolean?) {
        if (show == null) return
        if (show) {
            dialog.show()
        } else {
            dialog.dismiss()
        }
    }
}

然后在MainActivity添加這個(gè)Observer

vm.loading.observe(this, LoadingObserver(this))

效果:

加載對(duì)話(huà)框顯示

我們還可以將LoadingObserver注冊(cè)到BaseActivity

class BaseActivity : AppCompatActivity() {
    val loadingState = MutableLiveData<Boolean>()
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        loadingState.observe(this, LoadingObserver(this))
    }
}

然后在HomeVM中添加一個(gè)attachLoading方法

class HomeVM:ViewModel{
     fun attachLoading(otherLoadingState: MutableLiveData<Boolean>) {
        loading.observeForever {
            otherLoadingState.value = it
        }
    }
}

最終如果想要顯示進(jìn)度對(duì)話(huà)框,在BaseActivity到子類(lèi)中刃永,只需調(diào)用vm.attachLoading(loadingState)即可货矮。

分頁(yè)請(qǐng)求

分頁(yè)請(qǐng)求是另個(gè)一常用請(qǐng)求,它的請(qǐng)求狀態(tài)就比刷新數(shù)據(jù)多了幾種斯够。以wanandroid首頁(yè)文章列表api為例囚玫,我們?cè)?code>HomeVM中加入page,refreshing,moreLoadinghasMore變量控制分頁(yè)請(qǐng)求

private val page = MutableLiveData<Int>() //分頁(yè)數(shù)據(jù)
val refreshing = MutableLiveData<Boolean>()//下拉刷新?tīng)顟B(tài)
val moreLoading = MutableLiveData<Boolean>()//上拉加載更多狀態(tài)
val hasMore = MutableLiveData<Boolean>()//是否還有更多數(shù)據(jù)
private val articleList = Transformations.switchMap(page) {
    api.articleList(it)
}

val articlePage = Transformations.map(articleList) {
    refreshing.value = false
    moreLoading.value = false
    hasMore.value = !(it?.data?.over ?: false)
    it.data
}

fun loadMore() {
    page.value = (page.value ?: 0) + 1
    moreLoading.value = true
}

fun refresh() {
    loadBanner()
    page.value = 0
    refreshing.value = true
}

SmartRefreshLayout作為分頁(yè)組件读规,來(lái)實(shí)現(xiàn)WanAndroid首頁(yè)文章列表數(shù)據(jù)的展示抓督。

綁定SmartRefreshLayout屬性和事件

通過(guò)@BindingAdapter注解,將綁定SmartRefreshLayout屬性和事件封裝一樣束亏,便于我們?cè)诓季治募ㄟ^(guò)databinding控制它铃在。
新建一個(gè)CommonBinding.kt文件,注意在gradle中引入kotlin-kapt

@BindingAdapter(value = ["refreshing", "moreLoading", "hasMore"], requireAll = false)
fun bindSmartRefreshLayout(
    smartLayout: SmartRefreshLayout,
    refreshing: Boolean,
    moreLoading: Boolean,
    hasMore: Boolean

) {
    if (!refreshing) smartLayout.finishRefresh()
    if (!moreLoading) smartLayout.finishLoadMore()
    smartLayout.setEnableLoadMore(hasMore)
}

@BindingAdapter(value = ["onRefreshListener", "onLoadMoreListener"], requireAll = false)
fun bindListener(
    smartLayout: SmartRefreshLayout,
    refreshListener: OnRefreshListener?,
    loadMoreListener: OnLoadMoreListener?
) {
    smartLayout.setOnRefreshListener(refreshListener)
    smartLayout.setOnLoadMoreListener(loadMoreListener)
}

然后在布局中使用

<layout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        xmlns:tools="http://schemas.android.com/tools">

    <data>

        <variable
                name="vm"
                type="io.github.iamyours.wandroid.ui.home.HomeVM"/>
    </data>

    <com.scwang.smartrefresh.layout.SmartRefreshLayout
            android:id="@+id/refreshLayout"
            android:layout_width="match_parent"
            app:onRefreshListener="@{()->vm.refresh()}"
            app:refreshing="@{vm.refreshing}"
            app:moreLoading="@{vm.moreLoading}"
            app:hasMore="@{vm.hasMore}"
            app:onLoadMoreListener="@{()->vm.loadMore()}"
            android:layout_height="match_parent">

        <androidx.core.widget.NestedScrollView
                android:layout_width="match_parent"
                android:layout_height="match_parent">

            <LinearLayout
                    android:layout_width="match_parent"
                    android:orientation="vertical"
                    android:layout_height="wrap_content">

                <cn.bingoogolapple.bgabanner.BGABanner
                        android:id="@+id/banner"
                        android:layout_width="match_parent"
                        android:layout_height="140dp"
                        app:banner_indicatorGravity="bottom|right"
                        app:banner_isNumberIndicator="true"
                        app:banner_pointContainerBackground="#0000"
                        app:banner_transitionEffect="zoom"/>

                <androidx.recyclerview.widget.RecyclerView
                        android:id="@+id/recyclerView"
                        android:layout_width="match_parent"
                        android:layout_marginTop="5dp"
                        tools:listitem="@layout/item_article"
                        android:layout_height="wrap_content"/>
            </LinearLayout>
        </androidx.core.widget.NestedScrollView>
    </com.scwang.smartrefresh.layout.SmartRefreshLayout>

</layout>

分頁(yè)實(shí)現(xiàn)

然后在MainActivity中完成RecyclerView的邏輯

class MainActivity : AppCompatActivity() {
    lateinit var binding: ActivityMainBinding
    private val adapter = ArticleAdapter()
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = DataBindingUtil.setContentView(this, R.layout.activity_main)
        val vm = ViewModelProviders.of(this).get(HomeVM::class.java)
        binding.lifecycleOwner = this
        binding.vm = vm
        binding.executePendingBindings()
        initBanner()
        initRecyclerView()
        binding.refreshLayout.autoRefresh()
    }

    private fun initRecyclerView() {
        binding.recyclerView.let {
            it.adapter = adapter
            it.layoutManager = LinearLayoutManager(this)
        }
        binding.vm?.articlePage?.observe(this, Observer {
            it?.run {
                if (curPage == 1) {
                    adapter.clearAddAll(datas)
                } else {
                    adapter.addAll(datas)
                }
            }
        })
    }

    private fun initBanner() {
       ...
    }
}

最終效果:


wanandroid首頁(yè)數(shù)據(jù)

項(xiàng)目地址

https://github.com/iamyours/Wandroid

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末碍遍,一起剝皮案震驚了整個(gè)濱河市定铜,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌怕敬,老刑警劉巖揣炕,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異东跪,居然都是意外死亡畸陡,警方通過(guò)查閱死者的電腦和手機(jī)鹰溜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)丁恭,“玉大人曹动,你說(shuō)我怎么就攤上這事∩螅” “怎么了仁期?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵桑驱,是天一觀的道長(zhǎng)竭恬。 經(jīng)常有香客問(wèn)我,道長(zhǎng)熬的,這世上最難降的妖魔是什么痊硕? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮押框,結(jié)果婚禮上岔绸,老公的妹妹穿的比我還像新娘。我一直安慰自己橡伞,他們只是感情好盒揉,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著兑徘,像睡著了一般刚盈。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上挂脑,一...
    開(kāi)封第一講書(shū)人閱讀 51,292評(píng)論 1 301
  • 那天藕漱,我揣著相機(jī)與錄音,去河邊找鬼崭闲。 笑死肋联,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的刁俭。 我是一名探鬼主播橄仍,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼牍戚!你這毒婦竟也來(lái)了侮繁?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤翘魄,失蹤者是張志新(化名)和其女友劉穎鼎天,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體暑竟,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡斋射,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年育勺,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片罗岖。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡涧至,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出桑包,到底是詐尸還是另有隱情南蓬,我是刑警寧澤,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布哑了,位于F島的核電站赘方,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏弱左。R本人自食惡果不足惜窄陡,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望拆火。 院中可真熱鬧跳夭,春花似錦、人聲如沸们镜。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)模狭。三九已至颈抚,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間胞皱,已是汗流浹背邪意。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留反砌,地道東北人雾鬼。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像宴树,于是被迫代替她去往敵國(guó)和親策菜。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354