ListView更新界面的原理

本文探讨了ListView在删除或添加item时如何更新界面。通过观察者模式,当数据源改变,调用notifyDataSetChanged方法触发界面更新。我们深入源码,了解这个方法如何操作观察者,以及在setAdapter时如何创建和注册观察者,最终导致AdapterDataSetObserver的onChange方法调用,重新布局ListView。
摘要由CSDN通过智能技术生成

       ListView是我们经常使用的一个控件。那么,当我们删除或者添加ListView中的一个item时,界面是如何更新的呢?为了及时更新界面,ListView使用了观察者模式,在我们数据源发生改变时,我们调用了notifyDatasetchange方法,界面发生改变。下面,我们一起看看notifyDataSetChanged的源码实现。

        既然,更新界面需要调用notifyDataSetChange方法。那我们就首先看下这个方法的主要代码。

public abstract class BaseAdapter implements ListAdapter, SpinnerAdapter {
   
    private final DataSetObservable mDataSetObservable = new DataSetObservable();

   

    public void registerDataSetObserver(DataSetObserver observer) {
        mDataSetObservable.registerObserver(observer);
    }

    public void unregisterDataSetObserver(DataSetObserver observer) {
        mDataSetObservable.unregisterObserver(observer);
    }

    /**
     * Notifies the attached observers that the underlying data has been changed
     * and any View reflecting the data set should refresh itself.
     * 
     */
    public void notifyDataSetChanged() {
        mDataSetObservable.notifyChanged();
    }
}

   在该方法中实现了对观察者的添加和删除,以及通知观察者做出变化的方法。接下来在看看notifyChange方法。

public class DataSetObservable extends Observable<DataSetObserver> {
    /**
     * Invokes onChanged on each observer. Called when the data set being observed has
     * changed, and which when read contains the new state of the data.
     */
    public void notifyChanged() {
        synchronized(mObservers) {
            
            for (int i = mObservers.size() - 1; i >= 0; i--) {
                mObservers.get(i).onChanged();
            }
        }
    }

}

调用了这个方法后,会倒叙遍历观察者集合里面的观察者做出响应。

当我们调用Adapter的setAdapter方法时候,会创建观察者并添加进观察者集合。

 public void setAdapter(ListAdapter adapter) {
        // 如果已经有了一个adapter,那么先注销该Adapter对应的观察者
        if (mAdapter != null && mDataSetObserver != null) {
            mAdapter.unregisterDataSetObserver(mDataSetObserver);
        }

        // 代码省略

        super.setAdapter(adapter);

        if (mAdapter != null) {
            mAreAllItemsSelectable = mAdapter.areAllItemsEnabled();
            mOldItemCount = mItemCount;
            // 获取数据的数量
            mItemCount = mAdapter.getCount();
            checkFocus();
            // 注意这里 : 创建一个一个数据集观察者
            mDataSetObserver = new AdapterDataSetObserver();
            // 将这个观察者注册到Adapter中,实际上是注册到DataSetObservable中
            mAdapter.registerDataSetObserver(mDataSetObserver);

        } else {
           
        }

        requestLayout();
    }

在该方法中,创建了观察者并注册进被观察者集合中。

AdapterDataSetObserver继承自AdapterView中的AdapterDataSetObserver(继承自AbsListView中的DataSetObserver),布局改变后调用onChange方法中的requestLayout方法重新布局。

class AdapterDataSetObserver extends AdapterView<ListAdapter>.AdapterDataSetObserver {
        @Override
        public void onChanged() {
            super.onChanged();
            if (mFastScroll != null) {
                mFastScroll.onSectionsChanged();
            }
        }

        @Override
        public void onInvalidated() {
            super.onInvalidated();
            if (mFastScroll != null) {
                mFastScroll.onSectionsChanged();
            }
        }
    }

class AdapterDataSetObserver extends DataSetObserver {

        private Parcelable mInstanceState = null;
  
        @Override
        public void onChanged() {
            mDataChanged = true;
            mOldItemCount = mItemCount;
          
            mItemCount = getAdapter().getCount();

            // Detect the case where a cursor that was previously invalidated has
            // been repopulated with new data.
            if (AdapterView.this.getAdapter().hasStableIds() && mInstanceState != null
                    && mOldItemCount == 0 && mItemCount > 0) {
                AdapterView.this.onRestoreInstanceState(mInstanceState);
                mInstanceState = null;
            } else {
                rememberSyncState();
            }
            checkFocus();
            
            requestLayout();
        }

      

        public void clearSavedState() {
            mInstanceState = null;
        }
    }

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值