在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 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缓存及使用