适配器包括ArrayAdapter,SimpleAdapter和SimpleCursorAdapter。以及自定义适配器。
ArrayAdapter
数据来源:String[]和List
用法:1
publicclassArrayAdapterActivityextendsListActivity{
privateListView listView;
@Override
publicvoid onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
String[] strs ={"1","2","3","4","5"};
ArrayAdapter<String> adapter =
newArrayAdapter<String>(this,android.R.layout.simple_expandable_list_item_1,strs);
setListAdapter(adapter);
}
}
}
用法:2
publicclassMyListViewextendsActivity{
privateListView listView;
@Override
publicvoid onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
ArrayAdapter<String> adapter =
newArrayAdapter<String>(this, android.R.layout.simple_expandable_list_item_1,getData());
listView =newListView(this);
listView.setAdapter();
setContentView(listView);
}
privateList<String> getData(){
List<String> data =newArrayList<String>();
data.add("测试数据1");
data.add("测试数据2");
data.add("测试数据3");
data.add("测试数据4");
return data;
}
}
上面代码使用了Adapter(Context context, int resourcefulness, List objects)来装配数据,要装配这些数据就需要一个连接List View视图对象和数组数据的适配器来两者的适配工作,Adapter的构造需要三个参数,依次为this,布局文件(注意这里的布局文件描述的是列表的每一行的布局,android.R.layout.simple_list_item_1是系统定义好的布局文件只显示一行文字,数据源(一个List集合)。同时用adapter()完成适配的最后工作。
SimpleAdapter
SimpleAdapter的扩展性最好,可以定义各种各样的布局出来,可以放上ImageView(图片),还可以放上Button(按钮),CheckBox(复选框)等等。下面的代码都直接继承了ListActivity,ListActivity和普通的Activity没有太大的差别,不同就是对显示ListView做了许多优化,方面显示而已。
案例
<?xml version="1.0" encoding="utf-8"?>
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<ImageView
android:id="@+id/img"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="5dp"/>
<TextView
android:id="@+id/title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#ffffff"
android:textSize="20sp"/>
</LinearLayout>
publicclassSimpleAdapterActivityextendsListActivity{
@Override
publicvoid onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
SimpleAdapter adapter =
newSimpleAdapter(this, getData(), R.layout.simple,
newString[]{"title","img"},
newint[]{ R.id.title, R.id.img }
);
setListAdapter(adapter);
}
privateList<Map<String,Object>> getData(){
//map.put(参数名字,参数值)
List<Map<String,Object>> list =newArrayList<Map<String,Object>>();
Map<String,Object> map =null;
map =newHashMap<String,Object>();
map.put("title","摩托罗拉");
map.put("img", R.drawable.icon);
list.add(map);
map =newHashMap<String,Object>();
map.put("title","诺基亚");
map.put("img", R.drawable.icon);
list.add(map);
map =newHashMap<String,Object>();
map.put("title","三星");
map.put("img", R.drawable.icon);
list.add(map);
return list;
}
}
SimpleCursorAdapter
案例
publicclassSimpleCursorAdapterActivityextendsListActivity{
@Override
publicvoid onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
//获得一个指向系统通讯录数据库的Cursor对象获得数据来源
Cursor cur = getContentResolver().query(People.CONTENT_URI,null,null,null,null);
startManagingCursor(cur);
//实例化列表适配器
ListAdapter adapter =newSimpleCursorAdapter(this, android.R.layout.simple_list_item_1, cur,
newString[]{People.NAME},
newint[]{android.R.id.text1});
setListAdapter(adapter);
}
}
自定义Adapter
有时候,列表不光会用来做显示用,我们同样可以在在上面添加按钮。添加按钮首先要写一个有按钮的xml文件,然后自然会想到用上面的方法定义一个适配器,然后将数据映射到布局文件上。但是事实并非这样,因为按钮是无法映射的,即使你成功的用布局文件显示出了按钮也无法添加按钮的响应,这时就要研究一下ListView是如何现实的了,而且必须要重写一个类继承BaseAdapter。下面的示例将显示一个按钮和一个图片,两行字如果单击按钮将删除此按钮的所在行。并告诉你ListView究竟是如何工作的。
<?xml version="1.0" encoding="utf-8"?>
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="horizontal"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<ImageView
android:id="@+id/img"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="5px"/>
<LinearLayout
android:orientation="vertical"
android:layout_width="wrap_content"
android:layout_height="wrap_content">
<TextView
android:id="@+id/title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#FFFFFFFF"
android:textSize="22px"/>
<TextView
android:id="@+id/info"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#FFFFFFFF"
android:textSize="13px"/>
</LinearLayout>
<Button
android:id="@+id/view_btn"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/s_view_btn"
android:layout_gravity="bottom|right"/>
</LinearLayout>
publicclassMyListView4extendsListActivity{
privateList<Map<String,Object>> mData;
@Override
publicvoid onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
mData = getData();
MyAdapter adapter =newMyAdapter(this);
setListAdapter(adapter);
}
privateList<Map<String,Object>> getData(){
List<Map<String,Object>> list =newArrayList<Map<String,Object>>();
Map<String,Object> map =null;
map =newHashMap<String,Object>();
map.put("title","G1");
map.put("info","google 1");
map.put("img", R.drawable.i1);
list.add(map);
map =newHashMap<String,Object>();
map.put("title","G2");
map.put("info","google 2");
map.put("img", R.drawable.i2);
list.add(map);
map =newHashMap<String,Object>();
map.put("title","G3");
map.put("info","google 3");
map.put("img", R.drawable.i3);
list.add(map);
return list;
}
// ListView 中某项被选中后的逻辑
@Override
protectedvoid onListItemClick(ListView l,View v,int position,long id){
Log.v("MyListView4-click",(String)mData.get(position).get("title"));
}
//listview中点击按键弹出对话框
publicvoid showInfo(){
newAlertDialog.Builder(this)
.setTitle("我的listview")
.setMessage("介绍...")
.setPositiveButton("确定",newDialogInterface.OnClickListener(){
@Override
publicvoid onClick(DialogInterface dialog,int which){
}
})
.show();
}
/** 当Listie有大量的数据需要加载的时候,会占据大量内存,影响性能,这时候 要按需填充并重新使用view来减少对象的创建
*最快的方式是定义一个Pewholder,将convex的tag设置为Pewholder,不为空时重新使用即可
*/
publicfinalclassViewHolder{
publicImageView img;
publicTextView title;
publicTextView info;
publicButton viewBtn;
}
publicclassMyAdapterextendsBaseAdapter{
privateLayoutInflater mInflater;
publicMyAdapter(Context context){
this.mInflater =LayoutInflater.from(context);
}
@Override
publicint getCount(){
return mData.size();
}
@Override
publicObject getItem(int arg0){
returnnull;
}
@Override
publiclong getItemId(int arg0){
return0;
}
@Override
publicView getView(int position,View convertView,ViewGroup parent){
ViewHolder holder =null;
if(convertView ==null){
holder=newViewHolder();
convertView = mInflater.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);
holder.viewBtn =(Button)convertView.findViewById(R.id.view_btn);
convertView.setTag(holder);
}else{
holder =(ViewHolder)convertView.getTag();
}
holder.img.setBackgroundResource((Integer)mData.get(position).get("img"));
holder.title.setText((String)mData.get(position).get("title"));
holder.info.setText((String)mData.get(position).get("info"));
holder.viewBtn.setOnClickListener(newView.OnClickListener(){
@Override
publicvoid onClick(View v){
showInfo();
}
});
return convertView;
}
}
}
下面将对上述代码,做详细的解释
listView在开始绘制的时候,系统首先调用getCount()函数,根据他的返回值得到listView的长度,然后根据这个长度,调用getView()逐一绘制每一行。如果你的getCount()返回值是0的话,列表将不显示同样return 1,就只显示一行。
系统显示列表时,首先实例化一个适配器(这里将实例化自定义的适配器)。当手动完成适配时,必须手动映射数据,这需要重写getView()方法。系统在绘制列表的每一行的时候将调用此方法。
getView()有三个参数,position表示将显示的是第几行,covertView是从布局文件中inflate来的布局。我们用LayoutInflater的方法将定义好的vlist2.xml文件提取成View实例用来显示。然后将xml文件中的各个组件实例化(简单的findViewById()方法)。这样便可以将数据对应到各个组件上了。但是按钮为了响应点击事件,需要为它添加点击监听器,这样就能捕获点击事件。至此一个自定义的listView就完成了,现在让我们回过头从新审视这个过程。系统要绘制ListView了,他首先获得要绘制的这个列表的长度,然后开始绘制第一行,怎么绘制呢?调用getView()函数。在这个函数里面首先获得一个View(实际上是一个ViewGroup),然后再实例并设置各个组件,显示之。好了,绘制完这一行了。那再绘制下一行,直到绘完为止。在实际的运行过程中会发现listView的每一行没有焦点了,这是因为Button抢夺了listView的焦点,只要布局文件中将Button设置为没有焦点就OK了。