Listview详解,listview
Listview应该是最为常见的控件。对于大多数规则排列的界面,几乎都可以用ListView进行编写。对于单一界面来说,ListView既是最难的控件,又是使用最为频繁的控件。ListView 通常用于展示大量的数据,比如读取数据库中的数据。ListView优势也较为明显,比如显示大量数据时节约内存,自带ScrollView 的功能可以实现界面滚动等。ListView 控件的设计正好遵循MVC 设计模式。mode 数据数据模型,在ListView中具体可以指要显示到ListView 上的数据集合;view 视图用于展示数据,数据显示在每一个条目上;controller 控制层,数组中的数据是无法直接传递给 ListView 的,需要借助适配器把数据展示到控件上。因此有必要对ListView的使用进行一下总结。
Listview使用步骤并不复杂,无论是哪种场景下使用都可以归纳为以下几步:
1. 布局文件里申明ListView ,设置id号和其他相关参数
2. 在代码中通过findViewById方法找到listView控件
3. 设置适配器,可以单独定义出一个类继承相关的适配器,也可以创建一个适配器的匿名内部类。关键取决于代码的复杂程度。
4. 重写适配器中的方法。主要有两个,一个是getCount()表示listview的条目数。另一个是getView返回显示的View ,表示每一个条目的视图。getView()方法在每个子项被滚动到屏幕内的时候会被调用。
5. ListView列表项的点击事件 listView.setOnItemClickLinstener()
知道Listview的使用步骤后,还有很多需要注意的细节问题。接下来一一列举出来。
首先是适配器的类型选择。Android 中提供了很多适配器的实现类,通常情况下使用的都是BaseAdapter,而其他的适配器都是继承自BaseAdapter。但有些界面控件很少,利用ArrayAdapter或者SimpleAdapter就可以满足需求。所以在这里还是稍微介绍一下ArrayAdapter和SimpleAdapter的使用。文章最后给出一个学生信息管理系统的案例,用的适配器是BaseAdapter,所以这里就不再对BaseAdapter的使用再做讲解。
1. ArrayAdapter的使用
ArrayAdapter通常用于显示较为简单的数组和集合数据,界面较为简单,直接向ArrayAdapter添加相关的参数即可。
public ArrayAdapter(Context context, int resource, T[] objects) { init(context, resource, 0, Arrays.asList(objects)); }
第一个参数context表示上下文。 resource是布局文件的ID号,objects表示要显示的数组或List集合。
2. SimpleAdapter
使用SimpleAdapter 的数据一般都是HashMap构成的List集合,List 的每一个对象对应ListView 的每一行。HashMap 的每个键值数据映射到布局文件中对应id 的组件上。这样可以方便的显示图文显示的条目,通常设置界面就是这种形式。
public SimpleAdapter(Context context, List<? extends Map<String, ?>> data,int resource, String[] from, int[] to) { mData = data; mResource = mDropDownResource = resource; mFrom = from; mTo = to; mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); }
context表示上下文。data表示要显示的数据,用键值对的形式暂时存放数据。resource是布局文件的ID号。from表示Map 集合中key 的数组,to表示item 布局中控件的id。后两个参数均是字符转数组,表示的是多个Item条目。
以上两种适配器通常用于显示较为简单的条目内容,比如纯文本的显示、简单的文本和图片组合。复杂的条目结构需要用借助BaseaAadpter进行显示。
接着是对代码的优化。通常使用ListView时涉及到的数据不会太少,测试的时候几十上百条的数据可能看不出影响,但当数据达到成千上万条的时候,未优化的listView很有可能出现异常。因此代码优化是十分有必要的。这里介绍两处代码的优化处理,一个是convertView的复用,另一个是控件缓存机制。
优化一 listview的复用
convertView用于将之前加载好的布局进行缓存,以便之后可以进行复用。通过使用convertView 对创建的视图对象进行复用,可以节约减少内存消耗。对于少量相同形式的数据可以用LinearLayout代替显示,但是当数据增加到上千条、上万条的时候,快速滚动滑动条就会不断地生成新的TextView,对于内存的消耗过大,容易造成内存溢出。Listview自带上下滑动的功能,因此可以将滑出屏幕的convertView进行回收利用,每当一个item看不见的时候,那个item就可以被复用起来了,这样,ListView 始终保持创建的对象个数为屏幕显示的条目的个数加一。事实上,item的view对象没有真正的被垃圾回收器回收掉,而是重新将身上的数据给换掉了,看起来好像是出现了一个新的item。这样视觉上就是连续的滚动条了。
listview复用具体的做法是在 getView()方法中首先进行了判断convertView 的内容是否为空,如果convertView的内容为空则用layoutInflater 去加载布局,如果不为空则直接对 convertView 进行复用。这样就大大提高了ListView 的运行效率,在快速滚动的时候也可以表现出更好的性能,再也不用担心内存溢出了。
优化二 缓存控件的实例
获取布局文件中的控件,每次都要在getView()方法中调用 View的findViewById()方法来获取一次控件的实例,当需要多次关心控件时,就会创建多次,因此可以定义一个内部类,用于存放控件,最后将该类的对象存放在view的对象中。当 convertView 为空时,对控件的实例进行缓存,当 convertView 不为空的时候,从内部类中取出控件的实例,这样就减少了调用findViewById()方法的次数了,同样提高了代码的运行效率。
优化后的代码:
/** * 两点优化 * 1. 判断convertView是否为空 * 2. 内部类缓存控价 */ @Override public View getView(final int position, View convertView, ViewGroup parent) { View view; Viewholder viewholder; Student student = list.get(position); <