zoukankan      html  css  js  c++  java
  • 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. }  
  • 相关阅读:
    WTS_INFO_CLASS enumeration
    奇淫怪巧之给Delphi的PrintDialog增加一个页码选定范围打印的Edit
    Delphi的参数传递约定以及动态参数个数(转载笔记)
    C++中的模板那点事
    那些年我们一起学过的“排序算法”
    STL中的set容器的一点总结
    设计模式之备忘录模式(Memento)
    小程序员的趣味题(一)
    STL中的vector容器的一点总结
    STL中的list容器的一点总结
  • 原文地址:https://www.cnblogs.com/mamamia/p/9020033.html
Copyright © 2011-2022 走看看