随笔 - 262  文章 - 0  评论 - 5  阅读 - 20万

setHasFixedSize(true)的意义 (转)

RecyclerView setHasFixedSize(true)的意义
[java] view plain copy
 
  1. <span style="font-size:18px;">    /** 
  2.      * RecyclerView can perform several optimizations if it can know in advance that RecyclerView's 
  3.      * size is not affected by the adapter contents. RecyclerView can still change its size based 
  4.      * on other factors (e.g. its parent's size) but this size calculation cannot depend on the 
  5.      * size of its children or contents of its adapter (except the number of items in the adapter). 
  6.      * <p> 
  7.      * If your use of RecyclerView falls into this category, set this to {@code true}. It will allow 
  8.      * RecyclerView to avoid invalidating the whole layout when its adapter contents change. 
  9.      * 
  10.      * @param hasFixedSize true if adapter changes cannot affect the size of the RecyclerView. 
  11.      */  
  12.     public void setHasFixedSize(boolean hasFixedSize) {  
  13.         mHasFixedSize = hasFixedSize;  
  14.     }</span>  

注释说当知道Adapter内Item的改变不会影响RecyclerView宽高的时候,可以设置为true让RecyclerView避免重新计算大小。

设置为true,再调用notifyDataSetChanged(),发现大小重新计算了,看来理解出现错误了。还是再看一下哪些地方用到这个mHasFixedSize吧。

 

首先是onMeasure里用到,这个和自定义LayoutManager相关,先不管它。

剩下的就是triggerUpdateProcessor()方法了:

 

[java] view plain copy
 
  1. <span style="font-size:18px;">   void triggerUpdateProcessor() {  
  2.             if (POST_UPDATES_ON_ANIMATION && mHasFixedSize && mIsAttached) {  
  3.                 ViewCompat.postOnAnimation(RecyclerView.this, mUpdateChildViewsRunnable);  
  4.             } else {  
  5.                 mAdapterUpdateDuringMeasure = true;  
  6.                 requestLayout();  
  7.             }  
  8.         }</span>  

看一下triggerUpdateProcessor方法被哪些调用

 

onItemRangeChanged(),

onItemRangeInserted(),

onItemRangeRemoved(),

onItemRangeMoved()

这样看就很明白了,当调用Adapter的增删改插方法,最后就会根据mHasFixedSize这个值来判断需要不需要requestLayout();

再来看一下notifyDataSetChanged()执行的代码,最后是调用了onChanged,调用了requestLayout(),会去重新测量宽高。

 

[java] view plain copy
 
  1. @Override  
  2. public void onChanged() {  
  3.     assertNotInLayoutOrScroll(null);  
  4.     mState.mStructureChanged = true;  
  5.   
  6.     setDataSetChangedAfterLayout();  
  7.     if (!mAdapterHelper.hasPendingUpdates()) {  
  8.         requestLayout();  
  9.     }  
  10. }  

总结:当我们确定Item的改变不会影响RecyclerView的宽高的时候可以设置setHasFixedSize(true),并通过Adapter的增删改插方法去刷新RecyclerView,而不是通过notifyDataSetChanged()。(其实可以直接设置为true,当需要改变宽高的时候就用notifyDataSetChanged()去整体刷新一下)

 

 

[java] view plain copy
 
  1. public final void notifyItemChanged(int position) {  
  2.     mObservable.notifyItemRangeChanged(position, 1);  
  3. }  
  4.   
  5. public final void notifyItemChanged(int position, Object payload) {  
  6.     mObservable.notifyItemRangeChanged(position, 1, payload);  
  7. }  
  8.   
  9. public final void notifyItemRangeChanged(int positionStart, int itemCount) {  
  10.     mObservable.notifyItemRangeChanged(positionStart, itemCount);  
  11. }  
  12.   
  13. public final void notifyItemRangeChanged(int positionStart, int itemCount, Object payload) {  
  14.     mObservable.notifyItemRangeChanged(positionStart, itemCount, payload);  
  15. }  
  16.   
  17.   
  18. public final void notifyItemInserted(int position) {  
  19.     mObservable.notifyItemRangeInserted(position, 1);  
  20. }  
  21.   
  22.   
  23. public final void notifyItemMoved(int fromPosition, int toPosition) {  
  24.     mObservable.notifyItemMoved(fromPosition, toPosition);  
  25. }  
  26.   
  27. public final void notifyItemRangeInserted(int positionStart, int itemCount) {  
  28.     mObservable.notifyItemRangeInserted(positionStart, itemCount);  
  29. }  
  30.   
  31. public final void notifyItemRemoved(int position) {  
  32.     mObservable.notifyItemRangeRemoved(position, 1);  
  33. }  
  34.   
  35. public final void notifyItemRangeRemoved(int positionStart, int itemCount) {  
  36.     mObservable.notifyItemRangeRemoved(positionStart, itemCount);  
  37. }  
posted on   wp7ers  阅读(2084)  评论(2编辑  收藏  举报
编辑推荐:
· [.NET]调用本地 Deepseek 模型
· 一个费力不讨好的项目,让我损失了近一半的绩效!
· .NET Core 托管堆内存泄露/CPU异常的常见思路
· PostgreSQL 和 SQL Server 在统计信息维护中的关键差异
· C++代码改造为UTF-8编码问题的总结
阅读排行:
· 【.NET】调用本地 Deepseek 模型
· CSnakes vs Python.NET:高效嵌入与灵活互通的跨语言方案对比
· DeepSeek “源神”启动!「GitHub 热点速览」
· 我与微信审核的“相爱相杀”看个人小程序副业
· Plotly.NET 一个为 .NET 打造的强大开源交互式图表库
< 2025年2月 >
26 27 28 29 30 31 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 1
2 3 4 5 6 7 8

点击右上角即可分享
微信分享提示