ListView控件是android开发中最常用的控件之一。使用过程中经常配合Adapter类使用,listview是直接与手机使用者沟通的控件;而adapter是开发者将数据展现在listview上的桥梁,根据不同的数据使用adapter很关键。listview的类结构。
java.lang.Object | |||||
↳ | android.view.View | ||||
↳ | android.view.ViewGroup | ||||
↳ | android.widget.AdapterView<T extends android.widget.Adapter> | ||||
↳ | android.widget.AbsListView | ||||
↳ | android.widget.ListView |
(1)先来一个最简单的热身。listview中显示最简单的文本信息。这里我们用到arrayadapter这个类,它继承自baseadapter
ArrayAdapter
extends BaseAdapterimplements Filterable
activity_main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<ListView
android:id="@+id/listView"
android:layout_width="fill_parent"
android:layout_height="wrap_content" />
</LinearLayout>
list_item.xml
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/num"
android:layout_width="fill_parent"
android:layout_height="wrap_content" />
MainActivity.java
package com.example.test;
import java.util.ArrayList;
import android.app.Activity;
import android.os.Bundle;
import android.widget.ArrayAdapter;
import android.widget.ListView;
public class MainActivity extends Activity {
private ListView listView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
listView = (ListView) findViewById(R.id.listView);
ArrayList<String> students = new ArrayList<String>();
students.add("张三");
students.add("李四");
students.add("王五");
ArrayAdapter<String> adapter = new ArrayAdapter<String>(
MainActivity.this, R.layout.list_item, students);
listView.setAdapter(adapter);
}
}
完成
(2)listview中显示两个的文本信息,并对点击事件监听。这里我们用到simpleadapter这个类,它继承自baseadapter
SimpleAdapter
extends BaseAdapterimplements Filterable
在activity_main.xml中
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<ListView
android:id="@+id/listView"
android:layout_width="fill_parent"
android:layout_height="wrap_content" />
</LinearLayout>
list_item.xml定义了一个listview的一行的内容,也就是要显示的控件。我们往listview中加入两个textView控件,用于显示名字和年级。
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="horizontal" >
<TextView
android:id="@+id/nameTxt"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
<TextView
android:id="@+id/gradeTxt"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:paddingLeft="20dp" />
</LinearLayout>
最后是MainActivity.java
package com.example.test;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ListView;
import android.widget.SimpleAdapter;
import android.widget.Toast;
public class MainActivity extends Activity {
private ListView listView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 找到控件
listView = (ListView) findViewById(R.id.listView);
// 为了方便,定义一个二维数组
String[][] students = new String[][] { { "张三", "一年级" },
{ "王五", "二年级" }, { "赵六", "三年级" }, { "孙七", "四年级" } };
// arraylist用于保存map,map就是以key-value的形式保存信息
ArrayList<Map<String, String>> arrayList = new ArrayList<Map<String, String>>();
for (int i = 0; i < students.length; i++) {
// 一个学生信息保存在一个map中,每次都要new一个map对象。
Map<String, String> map = new HashMap<String, String>();
// 将信息放入到map中
map.put("name", students[i][0].toString());
map.put("grade", students[i][1].toString());
// 将map放入到arraylist中。完成学生信息的存储。
arrayList.add(map);
// 用simpleadapter类,第一个参数上下文对象,第二个参数是list类的对象,就是我们保存了信息的arraylist,
// 第三个参数是layout,就是listview一行,第四个参数是map中的key,第五个参数是layout中的控件id
SimpleAdapter adapter = new SimpleAdapter(MainActivity.this,
arrayList, R.layout.list_item, new String[] { "name",
"grade" },
new int[] { R.id.nameTxt, R.id.gradeTxt });
// 进行适配
listView.setAdapter(adapter);
listView.setOnItemClickListener(new ListViewItemClickListener());
}
}
private class ListViewItemClickListener implements OnItemClickListener {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position,
long id) {
Toast.makeText(MainActivity.this, position + " is clicked",
Toast.LENGTH_SHORT).show();
}
}
}
完成
(3)在二的基础上加一个checkbox,点击一个button后显示出选定的项。这次重写baseadapter这个基类,正因为是基类,拥有更大的灵活性,方便我们往里面放各种控件。
首先是activity_main.xml,非常简单
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:layout_gravity="bottom"
android:orientation="vertical" >
<ListView
android:id="@+id/listView"
android:layout_width="fill_parent"
android:layout_height="400dp" />
<Button
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:onClick="btnClick" />
</LinearLayout>
然后是list_item.xml,里面相对于二多加了一个checkbox控件。也是非常简单的。
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="horizontal" >
<TextView
android:id="@+id/nameTxt"
android:layout_width="wrap_content"
android:layout_height="fill_parent"
android:gravity="center_vertical" />
<TextView
android:id="@+id/gradeTxt"
android:layout_width="wrap_content"
android:layout_height="fill_parent"
android:gravity="center_vertical"
android:paddingLeft="20dp" />
<CheckBox
android:id="@+id/checkBox"
android:layout_width="wrap_content"
android:layout_height="fill_parent"
android:checked="false"
android:focusable="false"
android:gravity="center_vertical"
android:paddingLeft="20dp" />
</LinearLayout>
然后我们要写一个继承baseadapter的类,我起的名字是MyAdapter,必须要实现他的方法,如getItem,getCount等,要实现的最重要的是getView这个方法,不管是arrayadapter还是simpleadapter,他们都继承自baseadapter,他们也都要实现getView这个方法,这个方法,我的理解是,android系统为了优化流畅度,采用了一种很智能的算法。例如,如果我们的屏幕长度能盛下10个listview,那么android是不会说你有1000条数据就给你把1000个数据都放到1000个listview中,那手机可能承受不了。android只给你开11个listview,多出来的那个用来轮转,也就是为系统缓冲做准备啦。你往上滑动屏幕,android就会在最下面的一行及时填充数据,往下划屏幕同理。这样不管有多少数据,也不会卡顿。这个getView就是干这个的。listview初始化要调用getView,当滑动屏幕也要调用。
package com.example.test;
import java.util.ArrayList;
import java.util.Map;
import android.content.Context;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.CheckBox;
import android.widget.CompoundButton;
import android.widget.TextView;
public class MyAdapter extends BaseAdapter {
private Context context;
private ArrayList<Map<String, String>> arrayList;
private ArrayList<Boolean> isSelected;
public MyAdapter(Context context, ArrayList<Map<String, String>> arrayList,
ArrayList<Boolean> isSelected) {
this.context = context;
this.arrayList = arrayList;
this.isSelected = isSelected;
}
// 重写的方法是必须实现的,否则达不到效果。
// listview的行数是与arraylist里面的数据量相等的,这里直接返回这个量
@Override
public int getCount() {
// TODO Auto-generated method stub
return arrayList.size();
}
// 获得一行里面的object对象,
@Override
public Object getItem(int position) {
// TODO Auto-generated method stub
return arrayList;
}
@Override
public long getItemId(int position) {
// TODO Auto-generated method stub
return position;
}
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
// TODO Auto-generated method stub
LayoutInflater layoutInflater = LayoutInflater.from(context);
if (convertView == null) {
convertView = layoutInflater.inflate(R.layout.list_item, null);
}
TextView nameTxt = ViewHolder.get(convertView, R.id.nameTxt);
TextView gradeTxt = ViewHolder.get(convertView, R.id.gradeTxt);
CheckBox checkBox = ViewHolder.get(convertView, R.id.checkBox);
nameTxt.setText(arrayList.get(position).get("name"));
gradeTxt.setText(arrayList.get(position).get("grade"));
checkBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView,
boolean isChecked) {
// TODO Auto-generated method stub
if (isChecked) {
isSelected.set(position, true);
} else {
isSelected.set(position, false);
}
}
});
// 下面这个方法保证listview在滚动的时候被选中的项不会乱掉
checkBox.setChecked(isSelected.get(position));
return convertView;
}
public static class ViewHolder {
public static <T extends View> T get(View convertView, int id) {
SparseArray<View> viewHolder = (SparseArray<View>) convertView
.getTag();
if (viewHolder == null) {
viewHolder = new SparseArray<View>();
convertView.setTag(viewHolder);
}
View childView = viewHolder.get(id);
if (childView == null) {
childView = convertView.findViewById(id);
viewHolder.put(id, childView);
}
return (T) childView;
}
}
}
关于ViewHolder的写法,我自己也不是特别懂。是看某个大神写的。先记住啦。另外,
checkBox.setChecked(isSelected.get(position));
这一句主要是保证在滑动屏幕的时候,listview中选中的项不会乱掉。可以试验如果不写这一句会有什么后果。个人理解是,因为用getView始终要有一个轮转的行,如果不时时刷新checkbox的状态,listview就会乱掉。
isSelected集合用来记录每行选中的状态。
MyAdapter写完后,MainActivity.java就简单了。基本跟二一样。
package com.example.test;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;
import android.app.Activity;
import android.app.AlertDialog;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;
public class MainActivity extends Activity {
private ListView listView;
private ArrayList<Boolean> isSelected;
private ArrayList<Map<String, String>> arrayList;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 找到控件
listView = (ListView) findViewById(R.id.listView);
// 为了方便,定义一个二维数组
String[][] students = new String[][] { { "张三", "一年级" },
{ "王五", "二年级" }, { "赵六", "三年级" }, { "孙七", "四年级" },
{ "张三", "一年级" }, { "王五", "二年级" }, { "赵六", "三年级" },
{ "孙七", "四年级" }, { "张三", "一年级" }, { "王五", "二年级" },
{ "赵六", "三年级" }, { "孙七", "四年级" }, { "张三", "一年级" },
{ "王五", "二年级" }, { "赵六", "三年级" }, { "孙七", "四年级" },
{ "张三", "一年级" }, { "王五", "二年级" }, { "赵六", "三年级" },
{ "孙七", "四年级" }, { "张三", "一年级" }, { "王五", "二年级" },
{ "赵六", "三年级" }, { "孙七", "四年级" }, { "张三", "一年级" },
{ "王五", "二年级" }, { "赵六", "三年级" }, { "孙七", "四年级" },
{ "张三", "一年级" }, { "王五", "二年级" }, { "赵六", "三年级" },
{ "孙七", "四年级" }, { "张三", "一年级" }, { "王五", "二年级" },
{ "赵六", "三年级" }, { "孙七", "四年级" } };
// arraylist用于保存map,map就是以key-value的形式保存信息
arrayList = new ArrayList<Map<String, String>>();
isSelected = new ArrayList<Boolean>();
for (int i = 0; i < students.length; i++) {
// 一个学生信息保存在一个map中,每次都要new一个map对象。
Map<String, String> map = new HashMap<String, String>();
// 将信息放入到map中
map.put("name", students[i][0].toString());
map.put("grade", students[i][1].toString());
// 将map放入到arraylist中。完成学生信息的存储。
arrayList.add(map);
isSelected.add(false);
}
MyAdapter adapter = new MyAdapter(MainActivity.this, arrayList,
isSelected);
// 进行适配
listView.setAdapter(adapter);
listView.setOnItemClickListener(new ListViewItemClickListener());
}
public void btnClick(View view) {
StringBuffer sb = new StringBuffer();
for (int i = 0; i < isSelected.size(); i++) {
if (isSelected.get(i)) {
sb.append(arrayList.get(i)).append("\n");
}
}
TextView textView = new TextView(MainActivity.this);
textView.setText("选中的项有\n" + sb.toString());
AlertDialog dialog = new AlertDialog.Builder(MainActivity.this)
.setView(textView).create();
dialog.show();
}
private class ListViewItemClickListener implements OnItemClickListener {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position,
long id) {
Toast.makeText(MainActivity.this, position + " is clicked",
Toast.LENGTH_SHORT).show();
}
}
}
为了检查滑动时listview里面的checkbox会不会乱掉,二维数组里面加入了更多数据。也许有人会问,为什么不用OnItemClickListener里面的parent.getChildAt(index)这个方法。当时我也很纳闷。通过查资料知道,这个方法获得的是从当前显示行开始数的index行。看不到的行取不到啦。不知道为什么不直接写一个能取到的API,还是我不知道能用的api。总之上面是一种解决办法啦。
通过测试是可行的。这种方法用处之一是目录的操作。例如播放器选择扫描音乐目录。