一踪宠、列表顯示三要素:
- ListView:用于顯示列表的view;
- Adapter: 將數(shù)據(jù)映射到ListView鬼佣;
- Data: 具體的將要被映射的字符串垛叨、圖片或其他基本組件;
二装畅、優(yōu)化原理:
- 只創(chuàng)建必要的view對象靠娱;
ListView加載數(shù)據(jù)的方式: 首先調(diào)用getCount()函數(shù),根據(jù)返回值得到listView的長度掠兄;然后根據(jù)這個長度像云,調(diào)用getView()獲取listView的每一項item所對應(yīng)的view。
如果每一個item都去創(chuàng)建新的對象蚂夕,必然會占據(jù)過多的內(nèi)存迅诬。 - 盡可能減少去執(zhí)行Layout的inflate操作;
Infalte操作會利用xml文件實例化相應(yīng)的View,屬于IO操作双抽,是耗時操作百框。 - 減少findViewById()操作;
通過findViewById()方法可以在一個xml文件中尋找view牍汹。這個方法會遞歸整個view樹铐维,以尋找與ID相匹配的view。
當(dāng)item中含有較復(fù)雜的布局時慎菲,頻繁的回調(diào)findViewById()方法無疑會影響滑動的性能嫁蛇。
三、優(yōu)化機制:
-
Recycler(實現(xiàn)原理1和原理2)
當(dāng)listView的item離開當(dāng)前的視圖時露该,對應(yīng)的view會被緩存在Recycler中睬棚。
這時會相應(yīng)的從下方生成一個item,而此時調(diào)用的getView()函數(shù)中的convertView參數(shù)就是緩存的view解幼。
所以listView性能優(yōu)化的關(guān)鍵在于重用convertView抑党。
viewHolder靜態(tài)類(實現(xiàn)原理3)
將view的xml中的元素封裝成ViewHolder靜態(tài)類,通過convertView的setTag將view與相應(yīng)的holder對象綁定在一起撵摆。避免大量的findViewById()操作底靠。
TagsUnlike IDs, tags are not used to identify views. Tags are essentially an extra piece of information that can be associated with a view. They are most often used as a convenience to store data related to views in the views themselves rather than by putting them in a separate structure.
(利用tag關(guān)聯(lián)view的額外信息)
四、優(yōu)化過程:
- 最初寫法
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View view = LayoutInflater.from(mContext).inflate(R.layout.listview_item, null);
ImageView imageView1 = (ImageView)view.findViewById(R.id.imageView1);
TextView textView2 = (TextView)view.findViewById(R.id.text2);
imageView1.setImageResource(R.drawable.ic_launcher);
textView2.setText(mData.get(position));
return view;
}
(存在問題:1.創(chuàng)建view過多;2.inflate操作過多;3.findViewById()調(diào)用次數(shù)過多)
- 重用convertView
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if(convertView == null){
convertView = LayoutInflater.from(mContext).inflate(R.layout.listview_item, null);
}
ImageView imageView1 = (ImageView)convertView.findViewById(R.id.imageView1);
TextView textView2 = (TextView)convertView.findViewById(R.id.text2);
imageView1.setImageResource(R.drawable.ic_launcher);
textView2.setText(mData.get(position));
return convertView;
}
利用Recycler機制特铝,成功解決問題1和問題2暑中;
- 利用ViewHolder靜態(tài)類
static class ViewHolder{
ImageView imageView1;
TextView textView1;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder;
if(convertView == null){
holder = new ViewHolder();
convertView = LayoutInflater.from(mContext).inflate(R.layout.listview_item, null);
holder.imageView1 = (ImageView)convertView.findViewById(R.id.imageView1);
holder.textView1 = (TextView)convertView.findViewById(R.id.text1);
convertView.setTag(holder);
}else{
holder = (ViewHolder)convertView.getTag();
}
holder.imageView1.setImageResource(R.drawable.ic_launcher);
holder.textView1.setText(mData.get(position));
return convertView;
}
利用viewHolder靜態(tài)類,成功解決問題3鲫剿;
(建議將ViewHolder寫為靜態(tài)類鳄逾,因為用static定義的靜態(tài)內(nèi)部類相對獨立,不能訪問所在類的方法和實體灵莲,占用資源更少雕凹。如果去掉static,可以彼此互相訪問實體,會浪費一些資源。)
- 利用依賴注入框架精簡代碼:
static class ViewHolder{
@InjectView(R.id.imageView1)
ImageView imageView1;
@InjectView (R.id.text1)
TextView textView1;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder;
if(convertView == null){
holder = new ViewHolder();
convertView = LayoutInflater.from(mContext).inflate(R.layout.listview_item, null);
convertView.setTag(holder);
}else{
holder = (ViewHolder)convertView.getTag();
}
holder.imageView1.setImageResource(R.drawable.ic_launcher);
holder.textView1.setText(mData.get(position));
return convertView;
}
依賴注入框架能幫你精簡應(yīng)用程序的代碼。代碼越少意味著出現(xiàn)問題或bug的次數(shù)也就越少请琳,從而可以把更多的精力花在項目中那些需要編寫或修改的部分粱挡,使得閱讀代碼更加容易。
[2015-03-18]