Paging库的使用

前言

之前写过一篇关于Paging的文章,结合Room数据库使用的

Paging

这篇内容是关于请求网络数据的。Pging是一个分页加载库,其中组重要的部分就是PagedList。通过DataSource 获取的数据传递给PagedList对象;PagedList再通过PagedListAdapter显示到RecyclerView

其中DataSource分为三种:

  • PageKeyedDataSource:一般用于加载带有上下页的数据
  • ItemKeyedDataSource:一般用于上下文关联的数据,即通过上一个来获取下一个
  • PositionalDataSource:用于请求指定位置的一组数据

更详细的可以看官网介绍:custom-data-source

这里我在放下UP主的视频嘿嘿,大家可以直接去看

第47集 利用Paging实现加载更多(1)

第48集 利用Paging实现加载更多(2)

第49集 利用Paging实现加载更多(3)

正文

对了还有几点补充下:

本文内容都是基于UP主上一期视频的项目,有需要可以点此下载

该项目使用的是PageKeyedDataSource

网络请求库使用的是Volley

文中的Pixabaykey最好自行申请,不然多人使用过多会有上限

  1. 创建DataSource

    继承PageKeyedDataSource<Key, Value>,Key为Int,Value为请求的数据对象

    重写三个方法,分别是:初始化、加载上一页、加载下一页

    class PixabayDataSource(private val context: Context) : PageKeyedDataSource<Int, PhotoItem>() {
    
        private val keyWords = arrayOf("cat", "dog", "car", "beauty", "phone", "computer", "flower", "animal")
    
        override fun loadInitial(
            params: LoadInitialParams<Int>,
            callback: LoadInitialCallback<Int, PhotoItem>
        ) {
            val url =
                "https://pixabay.com/api/?key=17534539-6000088804093f98505e5666b&q=${keyWords.random()}&per_page=20&page=1"
            StringRequest(
                Request.Method.GET,
                url,
                Response.Listener {
                    val dataList = Gson().fromJson(it, Pixabay::class.java).hits.toList()
                    // onResult(List<Value> data, Key previousPageKey, Key nextPageKey)
                    callback.onResult(dataList, null, 2)
                },
                Response.ErrorListener {
                    Log.d("hello",it.toString())
                }
            ).also {
                VolleySingleton.getInstance(context).requestQueue.add(it)
            }
        }
    
        override fun loadAfter(params: LoadParams<Int>, callback: LoadCallback<Int, PhotoItem>) {
            val url =
                "https://pixabay.com/api/?key=17534539-6000088804093f98505e5666b&q=${keyWords.random()}&per_page=20&page=${params.key}"
            StringRequest(
                Request.Method.GET,
                url,
                Response.Listener {
                    val dataList = Gson().fromJson(it, Pixabay::class.java).hits.toList()
                    // onResult(List<Value> data, Key adjacentPageKey)
                    callback.onResult(dataList, params.key + 1)
                },
                Response.ErrorListener {
                    Log.d("hello",it.toString())
                }
            ).also {
                VolleySingleton.getInstance(context).requestQueue.add(it)
            }
        }
    
        override fun loadBefore(params: LoadParams<Int>, callback: LoadCallback<Int, PhotoItem>) {
        }
    }
    
  1. 创建DataSourceFactory

    继承DataSource.Factory<Key, Value>,重写create()方法

    class PixabayDataSourceFactory(private val context: Context) : DataSource.Factory<Int, PhotoItem>() {
        override fun create(): DataSource<Int, PhotoItem> {
            return PixabayDataSource(context)
        }
    }
    
  1. ViewModel中获取

    class GalleryViewModel(application: Application) : AndroidViewModel(application) {
        val pagedListPhoto = PixabayDataSourceFactory(application).toLiveData(1)
    }
    
  1. 最后在Activity/Fragment中调用即可

    galleryViewModel.pagedListPhoto.observe(viewLifecycleOwner, Observer {
     galleryAdapter.submitList(it)
    })
    

扩展一:网络状态

  1. DataSource中定义网络枚举类,同时设置一个MutableLiveData_networkStatus,供外部进行观察

    enum class NetworkStatus {
        LOADING,
        FAILED,
        COMPLETED
    }
    
    class PixabayDataSource(private val context: Context) : PageKeyedDataSource<Int, PhotoItem>() {
    
        private val keyWords = arrayOf("cat", "dog", "car", "beauty", "phone", "computer", "flower", "animal")
        private val _networkStatus = MutableLiveData<NetworkStatus>()
        val networkStatus: LiveData<NetworkStatus> = _networkStatus
    
        override fun loadInitial(
            params: LoadInitialParams<Int>,
            callback: LoadInitialCallback<Int, PhotoItem>
        ) {
            _networkStatus.postValue(NetworkStatus.LOADING)
            val url =
                "https://pixabay.com/api/?key=17534539-6000088804093f98505e5666b&q=${keyWords.random()}&per_page=20&page=1"
            StringRequest(
                Request.Method.GET,
                url,
                Response.Listener {
                    val dataList = Gson().fromJson(it, Pixabay::class.java).hits.toList()
                    callback.onResult(dataList, null, 2)
                },
                Response.ErrorListener {
                    _networkStatus.postValue(NetworkStatus.FAILED)
                    Log.d("hello",it.toString())
                }
            ).also {
                VolleySingleton.getInstance(context).requestQueue.add(it)
            }
        }
    
        override fun loadAfter(params: LoadParams<Int>, callback: LoadCallback<Int, PhotoItem>) {
            _networkStatus.postValue(NetworkStatus.LOADING)
            val url =
                "https://pixabay.com/api/?key=17534539-6000088804093f98505e5666b&q=${keyWords.random()}&per_page=20&page=${params.key}"
            StringRequest(
                Request.Method.GET,
                url,
                Response.Listener {
                    val dataList = Gson().fromJson(it, Pixabay::class.java).hits.toList()
                    callback.onResult(dataList, params.key + 1)
                },
                Response.ErrorListener {
                    _networkStatus.postValue(NetworkStatus.FAILED)
                    Log.d("hello",it.toString())
                }
            ).also {
                VolleySingleton.getInstance(context).requestQueue.add(it)
            }
        }
    
        override fun loadBefore(params: LoadParams<Int>, callback: LoadCallback<Int, PhotoItem>) {
        }
    }
    
  1. DataSourceFactory中,我们依然使用LiveData进行观察

    private val _pixabayDataSOurce = MutableLiveData<PixabayDataSource>()
    val pixabayDataSOurce: LiveData<PixabayDataSource> = _pixabayDataSOurce
    override fun create(): DataSource<Int, PhotoItem> {
        return PixabayDataSource(context)
        .also { _pixabayDataSOurce.postValue(it) }
    }
    
  1. ViewModel中进行传递

    class GalleryViewModel(application: Application) : AndroidViewModel(application) {
    
        private val factory = PixabayDataSourceFactory(application)
        val pagedListPhoto = factory.toLiveData(1)
        val networkStatus = Transformations.switchMap(factory.pixabayDataSOurce, {it.networkStatus})
        fun resetQuery() {
            pagedListPhoto.value?.dataSource?.invalidate()
        }
    }
    

    这里说一下这个Transformations.switchMap()。它内部通过MediatorLiveData实现

    MediatorLiveData继承自MutableLiveData,作用就是观察其他LiveData的变化

    switchMap可以理解成:A派B去监听C,你只需要告知我C的信息,所以文中可以获得DataSOurce中的

networkStatus观察对象

  1. ViewModel观察即可

    galleryViewModel.networkStatus.observe(viewLifecycleOwner, Observer {
        Log.d("networkStatus", it.toString())
    })
    

    运行应用后断开网络连接,往下华东加载图片,查看控制台日志变化

扩展二:重连

当用户因为网络而无法继续加载时,需要提供用户一个重连的机会

  1. DataSource中保存初始化函数,

    var retry: (() -> Any)? = null // 类型为函数
    
    ... ...
    
    override fun loadInitial(
        params: LoadInitialParams<Int>,
        callback: LoadInitialCallback<Int, PhotoItem>
    ) {
        retry = null // 发起请求时清空retry
        _networkStatus.postValue(NetworkStatus.LOADING)
        val url =
        "https://pixabay.com/api/?key=17534539-6000088804093f98505e5666b&q=${keyWords.random()}&per_page=20&page=1"
        StringRequest(
            Request.Method.GET,
            url,
            Response.Listener {
                val dataList = Gson().fromJson(it, Pixabay::class.java).hits.toList()
                callback.onResult(dataList, null, 2)
            },
            Response.ErrorListener {
                retry = {loadInitial(params, callback)} // 请求失败保存
                _networkStatus.postValue(NetworkStatus.FAILED)
                Log.d("hello",it.toString())
            }
        ).also {
            VolleySingleton.getInstance(context).requestQueue.add(it)
        }
    }
    
    
  1. ViewModel中获取

    fun retry() {
        factory.pixabayDataSOurce.value?.retry?.invoke()
    }
    

    最后在Activity/Fragment中调用即可

扩展三:优雅重连

当加载列表时网络断开了,需要在列表底部添加一个重试按钮,更加人性化

  1. 创建底部布局gallery_footer.xml

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="30dp"
        android:gravity="center">
    
        <ProgressBar
            android:id="@+id/progressBar"
            style="?android:attr/progressBarStyle"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content" />
    
        <TextView
            android:id="@+id/textView"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="TextView" />
    </LinearLayout>
    
  1. 修改GalleryAdapter

    class GalleryAdapter(private val galleryViewModel: GalleryViewModel) :
        PagedListAdapter<PhotoItem, RecyclerView.ViewHolder>(DIFFCALLBACK) {
    
        private var hasFooter = false 
        private var networkStatus: NetworkStatus? = null
    
        // 供外部调用,更新网络状态
        fun updateNetworkStatus(networkStatus: NetworkStatus) {
            this.networkStatus = networkStatus
            // 第一次加载的时候不需要显示
            if (networkStatus == NetworkStatus.INITIAL_LOADING) {
                hideFooter()
            } else {
                showFooter()
            }
        }
    
        private fun hideFooter() {
            // r如果当前已存在,则进行移除
            if (hasFooter) {
                notifyItemRemoved(itemCount - 1)
            }
            hasFooter = false
        }
    
        private fun showFooter() {
            // 如果已存在,则改变状态
            if (hasFooter) {
                notifyItemChanged(itemCount - 1)
            } else {
                hasFooter = true
                notifyItemInserted(itemCount - 1)
            }
        }
    
        override fun getItemCount(): Int {
            return super.getItemCount() + if (hasFooter) 1 else 0
        }
    
        override fun getItemViewType(position: Int): Int {
            return if (hasFooter && position == itemCount - 1) R.layout.gallery_footer else R.layout.gallery_cell
        }
    
        override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
    
            return when (viewType) {
                R.layout.gallery_cell -> PhotoViewHolder.getInstance(parent).also { holder ->
                    holder.itemView.setOnClickListener {
                        Bundle().apply {
                            putInt("PHOTO_POSITION", holder.adapterPosition)
                            holder.itemView.findNavController()
                            .navigate(R.id.action_galleryFragment_to_pagerPhotoFragment, this)
                        }
                    }
                }
                else -> FooterViewHolder.getInstance(parent).also {
                    it.itemView.setOnClickListener {
                        galleryViewModel.retry()
                    }
                }
            }
        }
    
        override fun onBindViewHolder(holder: RecyclerView.ViewHolder, position: Int) {
            when (holder.itemViewType) {
                R.layout.gallery_footer -> (holder as FooterViewHolder)
                .bindWithNetworkStatus(networkStatus)
                else -> {
                    val photoItem = getItem(position) ?: return
                    (holder as PhotoViewHolder).bindWithPhotoItem(photoItem)
                }
            }
        }
    
        object DIFFCALLBACK : DiffUtil.ItemCallback<PhotoItem>() {
            override fun areItemsTheSame(oldItem: PhotoItem, newItem: PhotoItem): Boolean {
                return oldItem === newItem
            }
    
            override fun areContentsTheSame(oldItem: PhotoItem, newItem: PhotoItem): Boolean {
                return oldItem.photoId == newItem.photoId
            }
        }
    
    }
    
    // 将原先onCreateViewHolder以及onBindViewHolder中的部分抽离出来
    class PhotoViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
    
        companion object {
    
            fun getInstance(parent: ViewGroup): PhotoViewHolder {
                val view = LayoutInflater.from(parent.context)
                    .inflate(R.layout.gallery_cell, parent, false)
                return PhotoViewHolder(view)
            }
        }
    
        fun bindWithPhotoItem(photoItem: PhotoItem) {
            with(itemView) {
                shimmerLayoutCell.apply {
                    setShimmerColor(0x55FFFFFF)
                    setShimmerAngle(0)
                    startShimmerAnimation()
                }
                textViewUser.text = photoItem.photoUser
                textViewLikes.text = photoItem.photoLikes.toString()
                textViewFavorites.text = photoItem.photoFavorites.toString()
                imageView.layoutParams.height = photoItem.photoHeight
            }
            itemView.shimmerLayoutCell.apply {
                setShimmerColor(0x55FFFFFF)
                setShimmerAngle(0)
                startShimmerAnimation()
            }
            Glide.with(itemView)
                .load(photoItem.previewUrl)
                .placeholder(R.drawable.photo_placeholder)
                .listener(object : RequestListener<Drawable> {
                    override fun onLoadFailed(
                        e: GlideException?,
                        model: Any?,
                        target: Target<Drawable>?,
                        isFirstResource: Boolean
                    ): Boolean {
                        return false
                    }
    
                    override fun onResourceReady(
                        resource: Drawable?,
                        model: Any?,
                        target: Target<Drawable>?,
                        dataSource: DataSource?,
                        isFirstResource: Boolean
                    ): Boolean {
                        return false.also { 
                            itemView.shimmerLayoutCell?.stopShimmerAnimation()
                        }
                    }
    
                })
                .into(itemView.imageView)
        }
    
    }
    
    class FooterViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
    
        companion object {
    
            fun getInstance(parent: ViewGroup): FooterViewHolder {
                val view = LayoutInflater.from(parent.context)
                    .inflate(R.layout.gallery_footer, parent, false)
                // 
                (view.layoutParams as StaggeredGridLayoutManager.LayoutParams).isFullSpan = true
                return FooterViewHolder(view)
            }
        }
    
        // 根据网络加载不同状态
        fun bindWithNetworkStatus(networkStatus: NetworkStatus?) {
            with(itemView) {
                when (networkStatus) {
                    NetworkStatus.FAILED -> {
                        textView.text = "点击重试"
                        progressBar.visibility = View.GONE
                        isClickable = true
                    }
                    NetworkStatus.COMPLETED -> {
                        textView.text = "加载完毕"
                        progressBar.visibility = View.GONE
                        isClickable = false
                    }
                    else -> {
                        textView.text = "正在加载"
                        progressBar.visibility = View.VISIBLE
                        isClickable = false
                    }
                }
            }
        }
    }
    

    上面还需要对DataSource进行修改,添加一种网络:INITIAL_LOADING,代表初次加载,同时loadInitial中的网络状态改为INITIAL_LOADING

  1. 剩下就是加载的问题了,直接看代码

    class GalleryFragment : Fragment() {
        // 此处的viewmodel作用域提升到Activity
        private val galleryViewModel by activityViewModels<GalleryViewModel>()
    
        override fun onCreateView(
            inflater: LayoutInflater, container: ViewGroup?,
            savedInstanceState: Bundle?
        ): View? {
            // Inflate the layout for this fragment
            return inflater.inflate(R.layout.fragment_gallery, container, false)
        }
    
        override fun onOptionsItemSelected(item: MenuItem): Boolean {
            when (item.itemId) {
                R.id.swipeIndicator -> {
                    swipeLayoutGallery.isRefreshing = true
                    galleryViewModel.resetQuery()
                }
                R.id.retry -> {
                    galleryViewModel.retry()
                }
            }
    
            return super.onOptionsItemSelected(item)
        }
    
        override fun onCreateOptionsMenu(menu: Menu, inflater: MenuInflater) {
            super.onCreateOptionsMenu(menu, inflater)
            inflater.inflate(R.menu.menu, menu)
        }
    
        override fun onActivityCreated(savedInstanceState: Bundle?) {
            super.onActivityCreated(savedInstanceState)
            setHasOptionsMenu(true)
            val galleryAdapter = GalleryAdapter(galleryViewModel)
            recyclerView.apply {
                adapter = galleryAdapter
                layoutManager = StaggeredGridLayoutManager(2, StaggeredGridLayoutManager.VERTICAL)
            }
    
            galleryViewModel.pagedListPhoto.observe(viewLifecycleOwner, Observer {
                galleryAdapter.submitList(it)
                swipeLayoutGallery.isRefreshing = false
            })
    
            swipeLayoutGallery.setOnRefreshListener {
                galleryViewModel.resetQuery()
            }
    
            // 这里监听网络状态,并更新adapter
            galleryViewModel.networkStatus.observe(viewLifecycleOwner, Observer {
                Log.d("networkStatus", it.toString())
                galleryAdapter.updateNetworkStatus(it)
            })
    
        }
    }
    

    PagerPhotoFragment

    const val REQUEST_WRITE_EXTERNAL_STORAGE = 1
    
    class PagerPhotoFragment : Fragment() {
    
        // 同样提升权限
        val galleryViewModel by activityViewModels<GalleryViewModel>()
    
        override fun onCreateView(
            inflater: LayoutInflater, container: ViewGroup?,
            savedInstanceState: Bundle?
        ): View? {
            // Inflate the layout for this fragment
            return inflater.inflate(R.layout.fragment_pager_photo, container, false)
        }
    
        override fun onActivityCreated(savedInstanceState: Bundle?) {
            super.onActivityCreated(savedInstanceState)
            val adapter = PagerPhotoListAdapter()
            viewPager2.adapter = adapter
            //数据从原来通过bundle传递改为viewmodel进行观察
            galleryViewModel.pagedListPhoto.observe(viewLifecycleOwner, Observer {
                adapter.submitList(it)
                viewPager2.setCurrentItem(arguments?.getInt("PHOTO_POSITION") ?: 0, false)
            })
    
            viewPager2.registerOnPageChangeCallback(object : ViewPager2.OnPageChangeCallback() {
                override fun onPageSelected(position: Int) {
                    super.onPageSelected(position)
                    photoTag.text = getString(R.string.photo_tag, position + 1, galleryViewModel.pagedListPhoto.value?.size)
                }
            })
    
    
            saveButton.setOnClickListener {
                if (Build.VERSION.SDK_INT < 29 && ContextCompat.checkSelfPermission(
                        requireContext(),
                        Manifest.permission.WRITE_EXTERNAL_STORAGE
                    ) != PackageManager.PERMISSION_GRANTED
                ) {
                    requestPermissions(
                        arrayOf(Manifest.permission.WRITE_EXTERNAL_STORAGE),
                        REQUEST_WRITE_EXTERNAL_STORAGE
                    )
                } else {
                    viewLifecycleOwner.lifecycleScope.launch {
                        savePhoto()
                    }
                }
            }
        }
    
        override fun onRequestPermissionsResult(
            requestCode: Int,
            permissions: Array<out String>,
            grantResults: IntArray
        ) {
            super.onRequestPermissionsResult(requestCode, permissions, grantResults)
            when (requestCode) {
                REQUEST_WRITE_EXTERNAL_STORAGE -> {
                    if (grantResults.isNotEmpty() && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                        viewLifecycleOwner.lifecycleScope.launch {
                            savePhoto()
                        }
                    } else {
                        Toast.makeText(requireContext(), "存储失败", Toast.LENGTH_SHORT).show()
                    }
                }
            }
        }
    
        private suspend fun savePhoto() {
            withContext(Dispatchers.IO) {
                val holder =
                    (viewPager2[0] as RecyclerView).findViewHolderForAdapterPosition(viewPager2.currentItem)
                            as PagerPhotoViewHolder
                val bitmap = holder.itemView.pagerPhoto.drawable.toBitmap()
    
                val saveUri = requireContext().contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
                    ContentValues()
                )?: kotlin.run {
                    Toast.makeText(requireContext(), "存储失败", Toast.LENGTH_SHORT).show()
                    return@withContext
                }
                requireContext().contentResolver.openOutputStream(saveUri).use {
                    if (bitmap.compress(Bitmap.CompressFormat.JPEG,90,it)) {
                        MainScope().launch {Toast.makeText(requireContext(), "存储成功", Toast.LENGTH_SHORT).show() }
                    } else {
                        MainScope().launch { Toast.makeText(requireContext(), "存储失败", Toast.LENGTH_SHORT).show() }
                    }
                }
            }
        }
    }
    

结尾

基本内容就是这些了,这篇文章只是大致描述了过程,如果看不明白的最好还是看视频来的直观点

(好久好久没更新了,太忙了(太懒了))

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351