Android开发ListView控件使用总结。

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
可以看到,listview可以当做是viewgroup,来存放一些其他控件,例如checkbox或button等。

(1)先来一个最简单的热身。listview中显示最简单的文本信息。这里我们用到arrayadapter这个类,它继承自baseadapter

ArrayAdapter

extends  BaseAdapter

implements 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  BaseAdapter
implements  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。总之上面是一种解决办法啦。


通过测试是可行的。这种方法用处之一是目录的操作。例如播放器选择扫描音乐目录。



  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值