(二十六) Android中ListView学习总结
一、列表的显示需要三个元素:
1、listView:用来显示列表的View
2、适配器: 用来把数据映射到listView上
3、数据:具体的将被映射的字符串、图片、或者基本组件。
二、系统自带的常用列表适配器:ArrayAdapter,SimpleAdapter和SimpleCursorAdapter,这三种适配器的使用大家可学习下官网上面的使用或者自行百度谷歌,一堆DEMO!!!其中以ArrayAdapter最为简单,只能展示一行字。SimpleAdapter有最好的扩充性,可以自定义出各种效果。SimpleCursorAdapter可以认为是SimpleAdapter对数据库的简单结合,可以方便的把数据库的内容以列表的形式展示出来。
三、ListView加载数据原理
1、系统要绘制ListView了,他首先用getCount()函数得到要绘制的这个列表的长度,然后开始绘制第一行,怎么绘制呢?调用getView()函数。在这个函数里面首先获得一个View(这个看实际情况,如果是一个简单的显示则是View,如果是一个自定义的里面包含很多控件的时候它其实是一个ViewGroup),然后再实例化并设置各个组件及其数据内容并显示它。好了,绘制完这一行了。那 再绘制下一行,直到绘完为止。
2、 ListView 针对每个item,要求 adapter “返回一个视图” (getView),也就是说ListView在开始绘制的时候,系统首先调用getCount()函数,根据他的返回值得到ListView的长度,然后根据这个长度,调用getView()一行一行的绘制ListView的每一项。如果你的getCount()返回值是0的话,列表一行都不会显示,如果返回1,就只显示一行。返回几则显示几行。如果我们有几千几万甚至更多的item要显示怎么办?为每个Item创建一个新的View?不可能!!!实际上Android早已经缓存了这些视图。在程序中的getView(int position, View convertView, ViewGroup parent)中的convertView就是拖动ListView 的时候被回收掉的view对象的缓存。
3、高效率的getView()实现方式如下所示、推荐使用下面的方法
public class MultipleItemsList extends ListActivity { private MyCustomAdapter mAdapter; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mAdapter = new MyCustomAdapter(); for (int i = 0; i < 50; i++) { mAdapter.addItem("item " + i); } setListAdapter(mAdapter); } private class MyCustomAdapter extends BaseAdapter { private ArrayList mData = new ArrayList(); private LayoutInflater mInflater; public MyCustomAdapter() { mInflater = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE); } public void addItem(final String item) { mData.add(item); notifyDataSetChanged(); } @Override public int getCount() { return mData.size(); } @Override public String getItem(int position) { return mData.get(position); } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { System.out.println("getView " + position + " " + convertView);
View view = convertView; //convertView就是拖动ListView 的时候被回收掉的view对象的缓存,重用了convertView,很大程度上的减少了内存的消耗。 ViewHolder holder = null; if (convertView == null) { view = mInflater.inflate(R.layout.item1, null); holder = new ViewHolder(); holder.textView = (TextView)view.findViewById(R.id.text); view.setTag(holder); } else { holder = (ViewHolder)view.getTag(); } holder.textView.setText(mData.get(position)); return view; } } public static class ViewHolder { //ViewHolder为static,静态类只会在第一次加载时会耗费比较长时间,但是后面就可以很好帮助加载,同时保证了内存中只有一个
///ViewHolder,节省了内存空间
public TextView textView;
效率不高的代码如下所示:问题所在:每次在getView的时候,都需要重新的findViewById,重新找到控件,然后进行控件的赋值以及事件相应设置。这样其实在做重复的事情,因为的geiview中,其实包含有这些控件,而且这些控件的id还都是一样的,也就是其实只要在view中findViewById一次,后面无需要每次都要findViewById了。
public View getView(int position, View convertView, ViewGroup parent) {
// TODO Auto-generated method stub
ApkFileInfo apkFileInfo = apkFiles.get(position);
View view = null;
if (convertView == null) {
view = View.inflate(context, R.layout.apk_clear_item,null);
} else {
view = convertView;
}
iv_apk_clear_icon = (ImageView) view.findViewById(R.id.iv_apk_clear_icon);
tv_apk_clear_name = (TextView) view.findViewById(R.id.tv_apk_clear_name);
cb_apk_clear = (CheckBox) view.findViewById(R.id.cb_apk_clear);
tv_apk_clear_size = (TextView) view.findViewById(R.id.tv_apk_clear_size);
iv_apk_clear_icon.setBackgroundDrawable(apkFileInfo.getApk_icon());
tv_apk_clear_name.setText(apkFileInfo.getApk_name());
cb_apk_clear.setChecked(apkFileInfo.getIschecked());
tv_apk_clear_size.setText(TextFormater.getDataSize(apkFileInfo.getApk_file_size()));
return view;
}
四、setTag()和getTag()方法
首先我们要知道setTag方法是干什么的,SDK解释为
Tags
Unlike 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不像ID是用标示view的。Tag从本质上来讲是就是相关联的view的额外的信息。它们经常用来存储一些view的数据,这样做非常方便而不用存入另外的单独结构。
1. 对于使用了LayoutInflater对象进行View扩充的Tag的使用
在之前,在adapter中,我们在getView中是这么些的代码:
public View getView(int position, View convertView, ViewGroup parent) { ViewHolder holder =null; if (convertView ==null) { holder =new ViewHolder(); convertView =inflater.inflate(R.layout.vlist2,null); holder.img = (ImageView) convertView.findViewById(R.id.img); holder.title = (TextView) convertView.findViewById(R.id.title); holder.info = (TextView) convertView.findViewById(R.id.info); // setTag的妙用 convertView.setTag(holder); }else { // setTag的妙用 holder = (ViewHolder) convertView.getTag(); } ……略 }
首先我们要知道setTag方法是干什么的,他是给View对象的一个标签,标签可以是任何内容,我们这里把他设置成了一个对象,因为我们是把vlist2.xml的元素抽象出来成为一个类ViewHolder,用了setTag,这个标签就是ViewHolder实例化后对象的一个属性。我们之后对于ViewHolder实例化的对象holder的操作,都会因为java的引用机制而一直存活并改变convertView的内容,而不是每次都是去new一个。我们就这样达到的重用——我希望我说清楚了。如果有更简单的解释,请指教。
这是我们在Adapter中的使用,那么我们在这里不使用Tag标签会怎么样呢?
我们试想,如果我们不用Tag标签,那么我们的对象如何与convertView缓存结合并达到合理的效率利用?貌似答案并不明朗——所以使用Tag是比较明智的做法。
2. 对于没有使用LayoutInflater对象进行View扩充的Tag的使用。
if (convertView != null) {
view = convertView;
...
} else {
view = new Xxx(...);
...
}
这是我们的程序,我们看到,貌似没有用Tag——是的,当没有使用LayoutInflater进行View的扩充的时候,是没有必要用的,虽然也可以用。
3. 对于其他View的Tag使用
我们可以对所有的View对象进行操作,至于怎么用,就看作者怎么想的了,下面举例说一个View的子类button对于tag的一个使用。
直接贴代码了:
publicclass ButtonTagTestActivityextends Activityimplements OnClickListener {
/** Called when the activity is first created. */
@Override
publicvoid onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button button1 = (Button) findViewById(R.id.button1);
Button button2 = (Button) findViewById(R.id.button2);
Button button3 = (Button) findViewById(R.id.button3);
button1.setTag(1);
button2.setTag(2);
button3.setTag(3);
button1.setOnClickListener(this);
}
@Override
publicvoid onClick(View arg0) {
// TODO Auto-generated method stub
int tag = (Integer) arg0.getTag();
switch (tag) {
case 1: {
Toast.makeText(this,"我是button1", Toast.LENGTH_LONG).show();
break;
}
case 2: {
Toast.makeText(this,"我是button2", Toast.LENGTH_LONG).show();
break;
}
case 3: {
Toast.makeText(this,"我是button3", Toast.LENGTH_LONG).show();
break;
}
default: {
break;
}
}
}
}
Xml页面代码就不贴了。这个例子是点击界面上的3个button然后会显示用户点击的按钮。我们的程序是实现了页面全局监听,在监听前设置了每个button的tag,之后我们在switch的时候,使用getTag取出的标签来看是什么操作。
这样做的好处是可以将监听集中管理,提高代码的易读性——当然,这是我的自我理解。
看了这么多的实例,我想已经明白了Tag以及convertView。
对我们知道了Tag的作用就是设置标签,标签可以是任意玩意。
以及convertView是如何在程序中使代码运行变的效率的:利用缓存convertView尽可能少实例化同样结构体的对象;
五:此随笔结合了博客:http://blog.csdn.net/yelangjueqi/article/details/9666037和博客:http://mzh3344258.blog.51cto.com/1823534/889879
浙公网安备 33010602011771号