1

そのため、Room データベースを使用してページング ライブラリを使用して簡単なメモ機能を作成しましLiveDataたが、実際には何もページ付けされていません。おそらく、ページネーションが行われていることを確認するために正しくデバッグしていません。アドバイスをいただければ幸いです。

私のDAOで:

@Query("SELECT * FROM notes ORDER BY id ASC")
fun allNotes(): DataSource.Factory<Int, NoteEntity>

次に、を からにNotesRepository変換して、データ レイヤーの実装の詳細とアプリの残りの部分を分離します (私はクリーン アーキテクチャのファンです)。したがって、結果は次のとおりです。NoteEntityNoteDataSource.Factory<Int, Note>

override fun allNotes(): DataSource.Factory<Int, Note> =
        notesDao.allNotes().map { mapper.fromDb(it) }

私のViewModel場所PAGE_SIZEは20です:

val noteList: LiveData<PagedList<Note>> =
        LivePagedListBuilder(
                getNotesUseCase.allNotes(), PAGE_SIZE)

Noteは単純です:

data class Note(
    val id: Long = 0,
    val text: String
)

次に、メモを に表示していますRecyclerView

私のViewHolder内容:

class NoteViewHolder(parent: ViewGroup) : RecyclerView.ViewHolder(
    LayoutInflater.from(parent.context).inflate(R.layout.note_item, parent, false))  {
   private val idView = itemView.findViewById<TextView>(R.id.noteId)
   private val nameView = itemView.findViewById<TextView>(R.id.noteText)
   private var note: Note? = null

   /**
    * Items might be null if they are not paged in yet. PagedListAdapter will re-bind the
    * ViewHolder when Item is loaded.
    */
    fun bindTo(note: Note?) {
        this.note = note
        idView.text = note?.let { it.id.toString() } ?: ""
        nameView.text = note?.text ?: ""
    }
}

私のAdapter内容:

class NoteAdapter(
    private val clickListener: ClickListener) : PagedListAdapter<Note, NoteViewHolder>(diffCallback) {

override fun onBindViewHolder(holder: NoteViewHolder, position: Int) {
    Timber.d("Binding view holder at position $position")
    val note = getItem(position)

    with(holder) {
        bindTo(note)
        note?.let {
            itemView.setOnClickListener {
            clickListener(note)
        }
    }
}

}

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

companion object {
    /**
     * This diff callback informs the PagedListAdapter how to compute list differences when new
     * PagedLists arrive.
     */
    private val diffCallback = object : DiffUtil.ItemCallback<Note>() {
        override fun areItemsTheSame(oldItem: Note, newItem: Note): Boolean =
                oldItem.id == newItem.id

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

そして私のFragment内容:

override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
    super.onViewCreated(view, savedInstanceState)

    setupRecyclerView()

    viewModel = ViewModelProviders.of(this, noteListViewModelFactory).get(NoteListViewModel::class.java)
    viewModel.noteList.observe(this, Observer { pagedNoteList ->
        pagedNoteList?.let { render(pagedNoteList) }
    })
}

private fun render(pagedNoteList: PagedList<Note>) {
    recyclerViewAdapter.submitList(pagedNoteList)
}

RecyclerView のデータが正常に読み込まれ、各アイテムがバインドされるとデバッグ ログが表示されます。私の理解ではPagedListAdapter、ユーザーがスクロールするときにリクエストの新しいページを使用PagedListし、バックグラウンド スレッドで違いを計算して新しい s を処理します。

私のデータ ソースには 100 個のアイテムが含まれているため、ページ サイズが 20 の場合、平均的なサイズの電話でテストすると、データに対して 5 回の呼び出しが行われると予想されます。

これをデバッグして正しい呼び出し数を確認するにはどうすればよいですか? また、上記の実装では、これらのいずれか (PositionalDataSource、ItemKeyedDataSource、または PageKeyedDataSource) がデフォルトで使用されていますか?

4

1 に答える 1

1

私自身の質問に答える:

in RoomPositionalDataSourceがデフォルトで使用されているようです。

my 内に次のログ ステートメントを追加することで、Notes がどのようにチャンクでロードされているかを把握できましたNoteAdapter.onBindViewHolder()

Log.d("NoteAdapter", currentList)

currentListスーパークラスのプロパティですPagedListAdapter

/**
 * Returns the PagedList currently being displayed by the Adapter.
 * etc.
 */
@Nullable
public PagedList<T> getCurrentList() {
    return mDiffer.getCurrentList();
}

ユーザーがスクロールすると、新しい項目がビュー ホルダーにバインドされ、ログにはcurrentList反復処理中の内容が表示されます。

Nまた、ページ サイズを に設定すると、最初のリスト サイズには のカウントが含まれることにも注意してください3 * N。その後、ユーザーがそれらすべてをスクロールするたびに、Nより多くのアイテムが に配置されcurrentListます。これは、ページ リストを要求するときにデフォルトの構成を使用したためです。

val noteList: LiveData<PagedList<Note>> =
        LivePagedListBuilder(
                getNotesUseCase.allNotes(), PAGE_SIZE)
                .build()

代わりに、次のようにして initialLoadSizeHint を制御することもできました。

private val pagedListConfig = PagedList.Config.Builder()
        .setEnablePlaceholders(true)
        .setInitialLoadSizeHint(INITIAL_LOAD_SIZE)
        .setPageSize(PAGE_SIZE)
        .build()

val noteList = LivePagedListBuilder<Int,Note>(getNotesUseCase.allNotes(), pagedListConfig).build()

デフォルトでは、initialLoadSizeHintは PAGE_SIZE * 3 に等しくなります。

于 2018-09-19T23:37:36.313 に答える