天天看點

Android Paging分頁庫的學習(二)—— 結合Room資料庫進行分頁加載

Paging分頁庫的介紹

Paging分頁面是google推出的一個結合RecyclerView進行分頁加載資料的一個全新架構庫,主要是為了解決一次性加載大量資料而造成的資源浪費問題。通過分頁的方式,每次加載一頁資料,既可以加快界面的渲染,又可以減少對象等資源的建立消耗。具體可以看官網

分頁庫主要由以下三個部分組成

  • DataSource: 資料源,定義擷取資料的方式,有三種方式,分别是
    1. PageKeyedDataSource
     2. ItemKeyedDataSource 
     3. PositionalDataSource.  基于位置資訊進行資料的加載,和Room資料庫或者本地資料源一起搭配。
               
  • PagedListAdapter: 分頁庫擴充卡,繼承于RecyclerView的擴充卡,内部需要實作一個DiffUtil.ItemCallback差分器分析資料是否發生了改變。
  • PagedList: 定義分頁庫的配置,分别有預設加載資料大小,分頁資料大小等。并且通過PagedListAdapter将資料的變化進行更新。

一、通過Room資料庫進行分頁加載

(一)DataSource的生成

Room是google官方資料庫架構:Room,它是一個ORM架構。詳情可以檢視這裡。相對于本地資料源Android Paging分頁庫的學習(一)—— 結合本地資料進行分頁加載,使用Room生成DataSource.Factory 很簡單,隻需在查詢的時候傳回DataSource.Factory類型就可以,如下圖

Android Paging分頁庫的學習(二)—— 結合Room資料庫進行分頁加載

而Room在編譯期為我們生成具體的實作類的實作方法,如下

Android Paging分頁庫的學習(二)—— 結合Room資料庫進行分頁加載

而具體的邏輯實作在LimitOffsetDataSource裡面,我們接着往下走,

Android Paging分頁庫的學習(二)—— 結合Room資料庫進行分頁加載

可以看到LimitOffsetDataSource繼承于PositionalDataSource,并且可以看到其中有兩個查詢語句是關鍵,分别是

1. “SELECT COUNT(*) FROM ( ” + mSourceQuery.getSql() + ” )”; 擷取資料源的大小,這個很容易了解,就是得到了全部要加載資料的總量。

2. “SELECT * FROM ( ” + mSourceQuery.getSql() + ” ) LIMIT ? OFFSET ?”; 這是查詢資料表的方式,Limit是每次查詢的數量,對應着分頁的大小,Offset是查詢的偏移位置,對應着每一頁的起始位置。

由于我們是通過Room生成PositionalDataSource的,部分的原理就先講到這裡,具體的PositionalDataSource的使用留着在加載本地資料的時候再說明。詳情戳

(二)PagedListAdapter的實作

由于PagedListAdapter繼承自RecyclerView的擴充卡,是以實作起來并不難,隻是需要提供一個差分的實作用來進行資料的分析,代碼如下:

class ArticlePageAdapter : PagedListAdapter<ArticleEntity, ArticleViewHolder>(diffCallback) {
    override fun onBindViewHolder(holder: ArticleViewHolder, position: Int) {
        holder.bindTo(getItem(position))
    }

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ArticleViewHolder =
            ArticleViewHolder(parent)

    companion object {
        private val diffCallback = object : DiffUtil.ItemCallback<ArticleEntity>() {
            override fun areItemsTheSame(oldItem: ArticleEntity, newItem: ArticleEntity): Boolean =
                    oldItem.id == newItem.id

            override fun areContentsTheSame(oldItem: ArticleEntity, newItem: ArticleEntity): Boolean =
                    oldItem == newItem
        }
    }
}
           

(三)PagedList的配置

PagedList主要是設定分頁的大小,初始化加載的資料大小等配置。

val pagedListConfig =PagedList.Config.Builder().setEnablePlaceholders(true).setPageSize().setInitialLoadSizeHint().build()
 val pagedList = LivePagedListBuilder(articleDao.getAllByDataSource(), pagedListConfig).build()
           

通過以上代碼生成是一個帶LiveData的PagedList

(四)總結

配置好Room資料庫,生成DataSource負責資料來源, 接着實作PagedListAdapter負責UI的渲染,最後進行PagedList分頁的一些配置。生成一個帶LiveData的PagedList,一旦資料進行變化,便會通知pageAdapter調用submitList進行UI的更新

recycle_article.layoutManager = LinearLayoutManager(this)
   val pageAdapter = ArticlePageAdapter()
   recycle_article.adapter = pageAdapter 
   val articleDao = dataBase.articleDao()
   val pagedListConfig = PagedList.Config.Builder().setEnablePlaceholders(true).setPageSize().setInitialLoadSizeHint().build()
   val postList = LivePagedListBuilder(articleDao.getAllByDataSource(), pagedListConfig).build()
   postList.observe(this, Observer { pageAdapter.submitList(it)})
           

demo已經上傳,點選傳送門,如有疑惑或者錯誤,歡迎指出。