在android开发中Listview是一个很重要的组件,它以列表的形式根据数据的长自适应展示具体内容。
用户可以自由的定义listview每一列的布局,但当listview有大量的数据需要加载的时候,会占据大量内存,影响性能,这时候就需要按需填充并重新使用view来减少对象的创建。
ListView加载数据都是在
1 public View getView(int position, View convertView, ViewGroup parent) {
2
3 。。。。。。
4
5 }
方法中进行的(要自定义listview都需要重写listadapter:如 BaseAdapter,SimpleAdapter,CursorAdapter的等的getvView方法),
优化listview的加载速度就要让 convertView匹配列表类型,并最大程度上的重新使用convertView。
其中,getview的加载方法一般有以下三种加载方式:
1、最慢的加载方式是每一次都重新定义一个View载入布局,再加载数据
1 public View getView(int position, View convertView, ViewGroup parent) {
2
3 View item = mInflater.inflate(R.layout.list_item_icon_text, null);
4
5 ((TextView) item.findViewById(R.id.text)).setText(DATA[position]);
6
7 ((ImageView) item.findViewById(R.id.icon)).setImageBitmap((position & 1) == 1 ? mIcon1 : mIcon2);
8
9 return item;
10
11 }
2、正确的加载方式是当convertView不为空的时候直接重新使用convertView从而减少了很多不必要的View的创建,然后加载数据
1 public View getView(int position, View convertView, ViewGroup parent) {
2
3 if (convertView == null) {
4
5 convertView = mInflater.inflate(R.layout.item, parent, false);
6
7 }
8
9 ((TextView) convertView.findViewById(R.id.text)).setText(DATA[position]);
10
11 ((ImageView) convertView.findViewById(R.id.icon)).setImageBitmap((position & 1) == 1 ? mIcon1 : mIcon2);
12
13
14
15 return convertView;
16
17 }
3、最快的方式是定义一个ViewHolder,将convetView的tag设置为ViewHolder,不为空时重新使用即可
1 static class ViewHolder {
2
3 TextView text;
4
5 ImageView icon;
6
7 }
8
9
10
11 public View getView(int position, View convertView, ViewGroup parent) {
12
13 ViewHolder holder;
14
15 if (convertView == null) {
16
17 convertView = mInflater.inflate(R.layout.list_item_icon_text,parent, false);
18
19 holder = new ViewHolder();
20
21 holder.text = (TextView) convertView.findViewById(R.id.text);
22
23 holder.icon = (ImageView) convertView.findViewById(R.id.icon);
24
25 convertView.setTag(holder);
26
27 } else {
28
29 holder = (ViewHolder) convertView.getTag();
30
31 }
32
33 holder.text.setText(DATA[position]);
34
35 holder.icon.setImageBitmap((position & 1) == 1 ? mIcon1 :mIcon2);
36
37 return convertView;
38
39 }
三种方式加载效率对比如下图所示:
说明:上述三个例子代码摘自google 2010 I/O大会
当处理一些耗时的资源加载的时候需要做到以下几点,以使你的加载更快更平滑:
1. 适配器在界面主线程中进行修改
2. 可以在任何地方获取数据但应该在另外一个地方请求数据
3. 在主界面的线程中提交适配器的变化并调用notifyDataSetChanged()方法
===============================分割线=========================================
那么如果存在多个item样式如何处理呢??
大致思路就是创建多个viewholder,在getViewType的时候设置不同位置的item用不同的viewholder,
以下直接上代码:
1 class MyAdapter extends BaseAdapter{
2
3 Context mContext;
4 LinearLayout linearLayout = null;
5 LayoutInflater inflater;
6 TextView tex;
7 final int VIEW_TYPE = 3;
8 final int TYPE_1 = 0;
9 final int TYPE_2 = 1;
10 final int TYPE_3 = 2;
11
12 //各个布局的控件资源
13 class viewHolder1{
14 CheckBox checkBox;
15 TextView textView;
16 }
17 class viewHolder2{
18 TextView textView;
19 }
20 class viewHolder3{
21 ImageView imageView;
22 TextView textView;
23 }
24
25 public MyAdapter(Context context) {
26 // TODO Auto-generated constructor stub
27 mContext = context;
28 inflater = LayoutInflater.from(mContext);
29 }
30
31 @Override
32 public int getCount() {
33 // TODO Auto-generated method stub
34 return listString.size();
35 }
36
37 //每个convert view都会调用此方法,获得当前所需要的view样式
38 @Override
39 public int getItemViewType(int position) {
40 // TODO Auto-generated method stub
41 int p = position%6;
42 if(p == 0)
43 return TYPE_1;
44 else if(p < 3)
45 return TYPE_2;
46 else if(p < 6)
47 return TYPE_3;
48 else
49 return TYPE_1;
50 }
51
52 //返回样式的数量
53 @Override
54 public int getViewTypeCount() {
55 // TODO Auto-generated method stub
56 return 3;
57 }
58
59 @Override
60 public Object getItem(int arg0) {
61 // TODO Auto-generated method stub
62 return listString.get(arg0);
63 }
64
65 @Override
66 public long getItemId(int position) {
67 // TODO Auto-generated method stub
68 return position;
69 }
70
71 @Override
72 public View getView(int position, View convertView, ViewGroup parent) {
73 // TODO Auto-generated method stub
74 viewHolder1 holder1 = null;
75 viewHolder2 holder2 = null;
76 viewHolder3 holder3 = null;
77 int type = getItemViewType(position);
78
79
80 //无convertView,需要new出各个控件
81 if(convertView == null)
82 {
83 Log.e("convertView = ", " NULL");
84
85 //按当前所需的样式,确定new的布局
86 switch(type)
87 {
88 case TYPE_1:
89 convertView = inflater.inflate(R.layout.listitem1, parent, false);
90 holder1 = new viewHolder1();
91 holder1.textView = (TextView)convertView.findViewById(R.id.textview1);
92 holder1.checkBox = (CheckBox)convertView.findViewById(R.id.checkbox);
93 Log.e("convertView = ", "NULL TYPE_1");
94 convertView.setTag(holder1);
95 break;
96 case TYPE_2:
97 convertView = inflater.inflate(R.layout.listitem2, parent, false);
98 holder2 = new viewHolder2();
99 holder2.textView = (TextView)convertView.findViewById(R.id.textview2);
100 Log.e("convertView = ", "NULL TYPE_2");
101 convertView.setTag(holder2);
102 break;
103 case TYPE_3:
104 convertView = inflater.inflate(R.layout.listitem3, parent, false);
105 holder3 = new viewHolder3();
106 holder3.textView = (TextView)convertView.findViewById(R.id.textview3);
107 holder3.imageView = (ImageView)convertView.findViewById(R.id.imageview);
108 Log.e("convertView = ", "NULL TYPE_3");
109 convertView.setTag(holder3);
110 break;
111 }
112 }
113 else
114 {
115 //有convertView,按样式,取得不用的布局
116 switch(type)
117 {
118 case TYPE_1:
119 holder1 = (viewHolder1) convertView.getTag();
120 Log.e("convertView !!!!!!= ", "NULL TYPE_1");
121 break;
122 case TYPE_2:
123 holder2 = (viewHolder2) convertView.getTag();
124 Log.e("convertView !!!!!!= ", "NULL TYPE_2");
125 break;
126 case TYPE_3:
127 holder3 = (viewHolder3) convertView.getTag();
128 Log.e("convertView !!!!!!= ", "NULL TYPE_3");
129 break;
130 }
131 }
132
133 //设置资源
134 switch(type)
135 {
136 case TYPE_1:
137 holder1.textView.setText(Integer.toString(position));
138 holder1.checkBox.setChecked(true);
139 break;
140 case TYPE_2:
141 holder2.textView.setText(Integer.toString(position));
142 break;
143 case TYPE_3:
144 holder3.textView.setText(Integer.toString(position));
145 holder3.imageView.setBackgroundResource(R.drawable.icon);
146 break;
147 }
148
149
150 return convertView;
151 }
152
153 }
参考原文:listview加载性能优化ViewHolder
参考原文:listView中多个listItem布局时,convertView缓存及使用