ListView之BaseAdapter详解

ListView是安卓中使用最重要的控件之一。实现ListView可以通过多种Adapter来完成。本文以BaseAdapter为例讲述ListView的用法,重点在于分析getView方法。

首先,在主Activity里初始化listview,同时在主Activity里建立内部类继承BaseAdapter(当然也可以新建一个文件在外面单独建立一个Adapter类或者直接建立匿名内部类)。
代码如下
MainActivity.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
package com.example.ahtcfg24.listviewdemo;

import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends Activity {
private ListView listView;
private List<String> list;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initListView();

}

private void initListView() {
listView = (ListView) findViewById(R.id.listView);
list = new ArrayList<>();
list.add("item0");
list.add("item1");
list.add("item2");
list.add("item3");
list.add("item4");
list.add("item5");
list.add("item6");
list.add("item7");
list.add("item8");
list.add("item9");
list.add("item10");
list.add("item11");
list.add("item12");
listView.setAdapter(new MyListAdapter(this, list));

}


private class MyListAdapter extends BaseAdapter {
private Context context;
private List<String> list;

public MyListAdapter(Context context, List<String> list) {
this.context = context;
this.list = list;
}

/**
* @return 要绘制的View数目,不能大于实际存在资源总数
*/

@Override
public int getCount() {
return list.size();
}

/**
* 覆盖自AdapterView,此处不会自动调用
*/

@Override
public Object getItem(int position) {
return null;
}

/**
* 点击时调用
*/

@Override
public long getItemId(int position) {
return 0;
}

/**
* 通过传入的position,把需要绘制的View数据加工成我们想要的View,最后返回给父级容器使用
* 把数据和控件联系起来的关键方法,形成一个缓冲模型,屏幕上能显示的View都在内存里,
* 而把不能显示的放进缓冲区队列Recycle里,缓冲区占用总大小是一个屏幕能显示的所有View数量+2再整体
* 乘单个View所占的内存空间
*
* @param position 即将脱离屏幕边缘的位置,如果屏幕向上滑动就是最上端的View所在位置
* @param convertView 即将出缓冲区再度进入内存被重新利用的View,在屏幕未被占满时,该参数为空,
* 此时可以为其新建一个View对象,也可以从XML布局文件中inflate出来一个View对象,
* 当屏幕被占满,有View进入缓冲区时,这时该参数就是一个已经存在过但已经脱离屏幕
* 范围时的View对象
* @param parent item的父级容器,一般都为空
* @return 要显示的每个item的View
*/

@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder viewHolder;
if (convertView == null) {
convertView = LayoutInflater.from(context).inflate(R.layout.item, null);
viewHolder = new ViewHolder();
viewHolder.textView = (TextView) convertView.findViewById(R.id.textView);
convertView.setTag(viewHolder);
} else {
viewHolder = (ViewHolder) convertView.getTag();
}
viewHolder.textView.setText(list.get(position));
return convertView;
}
}

public class ViewHolder {
private TextView textView;
}
}

其中ViewHolder类以及对convertView的为空判断属于ListView的一种优化方法,详情见这里

布局如下
activity_main.xml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="5dp"
tools:context=".MainActivity">


<TextView
android:id="@+id/textView"
android:layout_width="match_parent"
android:layout_height="200dp"
android:background="#00ffcc"
android:text="这是个TextView"
android:textSize="50sp"
android:layout_centerHorizontal="true"
android:layout_alignParentTop="true"/>



<ListView
android:layout_below="@+id/textView"
android:layout_margin="10dp"
android:background="#999999"
android:id="@+id/listView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
/>

</RelativeLayout>

(本文系作者原创,转载请注明出处)