android listview优化,Android ListView 详解及其优化
一、基本原理
ListView 是一個顯示一列可滾動項目的視圖組。 系統使用 Adapter 自動將列表項目插入列表,適配器從來源(例如數組或數據庫查詢)提取內容,并將每個項目結果轉換為視圖放置到列表中。
列表數據的顯示需要 4 個元素:
ListView 視圖組;
用來把數據映射到 ListView 上的 Adapter;
需要展示的數據集;
數據展示的 View 模版。
ListView 只負責加載和管理視圖,核心實現是由 Adapter 類完成。
在 Adapter 里主要實現這四個函數:
獲取數據的個數
獲取 position 位置的數據
獲取 position 位置的數據 id,一般直接返回 position 即可
獲取 position 位置上的每項數據(Item View)視圖
在 getView 函數中用戶必須構建 Item View,然后將該 position 位置上數據綁定到 Item View。
1. 視圖復用機制
Android 采用視圖復用的形式避免創建過多的 Item View,提升性能,降低內存占有率。
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View view = null;
//有視圖緩存,復用視圖
if (convertView != null) {
view = convertView;
} else {
//重新加載視圖
}
//進行數據綁定
//返回Item View
return view;
}
position 表示該視圖是第幾項數據,convertView 表示緩存的 Item View,parent 表示該 Item View 的父視圖,對于 ListView 來說這個 parent 就是 ListView 本身。
數據量較大時,ListView 不會構建全部 Item View,ListView 只會構建足夠鋪滿屏幕所需的 Item View 個數,例如 8 個 Item View 足夠鋪滿屏幕,數據項有 1000 個,那么 Item View 可以只產生 8 個,即反復利用容器,變換里面的數據,這樣可以節約內存,提高效率。
2. Adapter(適配器)模式
適配器模式(有時候也稱包裝樣式或者包裝)將一個類的接口適配成用戶所期待的。
共有兩類適配器模式:
對象適配器模式
在這種適配器模式中,適配器容納一個它包裹的類的實例。在這種情況下,適配器調用被包裹對象的物理實體。
類適配器模式
這種適配器模式下,適配器繼承自已實現的類(一般多重繼承)。
ListView 要想使用 getView() 方法,但是不同的數據,不同的需求,就會有不同的 getView() 結果,所以 getView() 必須是可復寫的,那么就可以想到用適配器模式。
ListView 里面包含了一個 ListAdapter 的成員變量,實際上是 ListView 繼承了 AbsListView,ListAdapter 變量是在 AbsListView 中聲明的。
BaseAdapter 是實現了 ListAdapter 的,于是,我們自定義的 xxxAdapter 繼承了 BaseAdapter,其實就相當于間接實現了 ListAdapter,然后我們就可以復寫 getView() 方法。
ListView 就可以通過調用 setAdapter(ListAdapter adapter) 方法,將自定義的 Adapter 傳進去,調用我們復寫后的方法,從而得到想要的結果。
3. 觀察者模式
Adapter 內部有一個可觀察者類,ListView 作為它的其中一個觀察者。
ListView 的 setAdapter() 方法:
@Override
public void setAdapter(ListAdapter adapter) {
if (mAdapter != null && mDataSetObserver != null) {
mAdapter.unregisterDataSetObserver(mDataSetObserver);
}
resetList();
//清空視圖緩存mRecycler
mRecycler.clear();
if (mAdapter != null) {
......
mDataSetObserver = new AdapterDataSetObserver();
//將 mDataSetObserver 注冊到 adapter 中
mAdapter.registerDataSetObserver(mDataSetObserver);
......
} else {
......
}
requestLayout();
}
BaseAdapter 部分代碼:
public abstract class BaseAdapter implements ListAdapter, SpinnerAdapter {
private final DataSetObservable mDataSetObservable = new DataSetObservable();
public boolean hasStableIds() {
return false;
}
public void registerDataSetObserver(DataSetObserver observer) {
mDataSetObservable.registerObserver(observer);
}
public void unregisterDataSetObserver(DataSetObserver observer) {
mDataSetObservable.unregisterObserver(observer);
}
注冊觀測者實際上是調用了 DataSetObservable 對應的函數。DataSetObservable 擁有一個觀察者集合,當可觀察者發生變更時,就會通知觀察者做出相應的處理。
public abstract class Observable {
/**
* The list of observers. An observer can be in the list at most
* once and will never be null.
*/
protected final ArrayList mObservers = new ArrayList();
public void registerObserver(T observer) {
if (observer == null) {
throw new IllegalArgumentException("The observer is null.");
}
synchronized(mObservers) {
mObservers.add(observer);
}
}
Adapter 的數據源發生變化時,我們會調用 Adapter 的
notifyDataSetChanged() 函數,在該函數中又會調用
DataSetObservable 對象的 notifyChanged() 函數通知所有的觀察者數據發生了變化,使觀察者進行相應的操作。
public class DataSetObservable extends Observable {
public void notifyChanged() {
synchronized(mObservers) {
for (int i = mObservers.size() - 1; i >= 0; i--) {
mObservers.get(i).onChanged();
}
}
}
對于 ListView 來說,這個觀察者就是 AdapterDataSetObserver 對象,該類聲明在 AdapterView 類中,也是 ListView 中的一個父類。
class AdapterDataSetObserver extends DataSetObserver {
private Parcelable mInstanceState = null;
@Override
public void onChanged() {
mDataChanged = true;
mOldItemCount = mItemCount;
//獲取元素個數
mItemCount = getAdapter().getCount();
if (AdapterView.this.getAdapter().hasStableIds() && mInstanceState != null
&& mOldItemCount == 0 && mItemCount > 0) {
AdapterView.this.onRestoreInstanceState(mInstanceState);
mInstanceState = null;
} else {
rememberSyncState();
}
checkFocus();
//重新布局
requestLayout();
}
onChanged() 函數中,實際調用的是 AdapterView 或者 ViewGroup 類中的屬性或者函數完成功能。因此,AdapterDataSetObserver 只是在外層做了一下包裝,真正的核心功能是 ListView,確切地說應該是 AdapterView。因此,源碼里使用 AdapterDataSetObserver 對象作為觀察者并不違反 ListView 作為真正觀察者的意圖。
二、處理點擊事件
可以通過實現 AdapterView.OnItemClickListener 來響應 AdapterView 中每一項上的點擊事件。 例如:
// Create a message handling object as an anonymous class.
private OnItemClickListener mMessageClickedHandler = new OnItemClickListener() {
public void onItemClick(AdapterView parent, View v, int position, long id) {
// Do something in response to the click
}
};
listView.setOnItemClickListener(mMessageClickedHandler);
三、其他
GridView 與 ListView 相似, 同樣繼承 AbsListView,AbsListView 又是 AdapterView 的子類,所以原理機制基本相同,不同的是,GridView 通過網格布局形式展示。
同一個 Adapter 可以設置給多個的列表視圖,數據會同時顯示且根據數據變化而變化,這些列表視圖都作為該 Adapter 的觀察者,這可能也是為什么要使用觀察者模式的一個原因。
總結
以上是生活随笔為你收集整理的android listview优化,Android ListView 详解及其优化的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: VS2019 配置OpenGL
- 下一篇: RubyOnRails 学习网站