<span style="font-family: Arial, Helvetica, sans-serif; font-size: 12px; background-color: rgb(255, 255, 255);"> 今天用</span><span style="font-family: Arial, Helvetica, sans-serif; font-size: 14px; background-color: rgb(255, 255, 255);">Android中BaseAdapter做了一个数据集合的展示,自己把自己给坑了。所以自己很有必要搞清楚Adapter的原理。</span>
对于Adapter的原理有一个很形象的比喻:现在有4瓶水《农夫山泉、康师傅、冰红茶、脉动》,我们知道这个4个瓶子的形状都不一样,尤其是瓶口不一样。现在需要我们向一个大的康师傅的瓶子里倒水,要求水不泼洒。
肿么办?
于是漏斗解决了这个问题。那么这里的漏斗就像是Adapter,而四种不一样的瓶子就如Android中widget数据集控件《ListView、GridView、Gallery、Spinner...》现在要将数据投入到这个里面,那么借助Adapter肿么实现呢?
Android的官方提供了很多数据适配器,将数据绑定到指定的View试图中
这里只介绍BaseAdapter
--------------------
使用数据适配器实现数据列表中增加图片和两行文本以及一个按钮。下面是代码
Main.xml
<?xml version="1.0" encoding="UTF-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
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"
android:fastScrollEnabled="true" />
</LinearLayout>
Activity
package com.lol.huixin.control;
import android.os.Bundle;
import android.util.Log;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import android.app.Activity;
import android.app.AlertDialog;
import android.app.ListActivity;
import android.content.Context;
import android.content.DialogInterface;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;
public class BaseAdapterActivity extends Activity {
private ListView listView;
private List<Map<String,Object>> listMap;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_base_adapter);
listView=(ListView)findViewById(R.id.listView);
listMap=getData();
MyAdapter adapter=new MyAdapter(getApplicationContext());
listView.setAdapter(adapter);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.base_adapter, menu);
return true;
}
private List<Map<String, Object>> getData() {
List<Map<String, Object>> list = new ArrayList<Map<String, Object>>();
Map<String, Object> map = new HashMap<String, Object>();
map.put("title", "G1");
map.put("info", "google 1");
map.put("img", R.drawable.a1);
list.add(map);
map = new HashMap<String, Object>();
map.put("title", "G2");
map.put("info", "google 2");
map.put("img", R.drawable.a2);
list.add(map);
map = new HashMap<String, Object>();
map.put("title", "G3");
map.put("info", "google 3");
map.put("img", R.drawable.a3);
list.add(map);
return list;
}
public static class ViewHolder{
public ImageView img;
public TextView title;
public TextView info;
public Button viewBtn;
}
public class MyAdapter extends BaseAdapter{
//LayoutInflater是用来寻找layout下xml布局文件,并且实例化
// findViewById()是找具体xml下具体的widget控件(Button TextView)
public LayoutInflater myInflater;
public MyAdapter(Context context){
根据context上下文加载布局,这里的是getApplicationContext()本身,即this
this.myInflater=LayoutInflater.from(context);
}
@Override
public int getCount() {
//在此适配器中所代表的数据集中的条目数
return listMap.size();
}
@Override
public Object getItem(int position) {
//获取数据集中与指定索引对应的数据项
return position;
}
@Override
public long getItemId(int position) {
//获取在列表中与指定索引对应的行id
return position;
}
//convertView就是每一Item在Recyler之前的布局视图
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder=null;
if(convertView==null){
//如果缓存convertView为空,则需要创建View
holder=new ViewHolder();
//这句话很关键
convertView=myInflater.inflate(R.layout.list_items, 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);
//将设置好的布局保存到缓存中,并将其设置在Tag里,以便后面方便取出Tag
convertView.setTag(holder);
}else{
holder=(ViewHolder)convertView.getTag();
}
Log.i("positon",String.valueOf(position)+"\t"+listMap.get(position).get("title"));
holder.img.setBackgroundResource((Integer)listMap.get(position).get("img"));
holder.title.setText((String)listMap.get(position).get("title"));
holder.info.setText((String)listMap.get(position).get("info"));
holder.viewBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
showInfo();
}
});
return convertView;
}
}
/**
* listview中点击按键弹出对话框
*/
public void showInfo(){
new AlertDialog.Builder(this).setTitle("我的listview").setMessage("介绍...").setPositiveButton("确定", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {}
}).show();
}
}
list_items.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns: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>
效果是这样:
我们来想为什么,要这么写?
虽然每一行代码都有注释,但是为啥这么一个简单的功能,要写这么多代码,脑壳有坑麽?
1.首先BaseAdapter这个适配器中的getView()方法中的是遍历每一个listMap中每一个items,也就需要我们在每一个item选项中创建图片、文本、按钮试图控件,需要独立一个试图集来存放这些东西,为啥呢,因为getView()方法是一个伪迭代器,他一直在遍历集合,如果我们每次都用findViewById方法去寻找试图组件,肯定是损耗内存。
2.一般来讲当我们寻找通过Activity创建的xml时,只需要通过findViewById,但是如果我们需要寻找独立创建的xml文件时,特别是在每一个遍历集合Item中寻找时,那么此时就需要判断缓存中是否存在我们寻找的xml,不存在就创建
3.为啥需要tag,一般来讲我们使用listView试图显示数据时,数据量很大,就涉及到分页翻取,在Android中,我不能每次翻页时,都从数据库中查询,那么会很占资源,而在Adapter适配器中的convertView中恰恰可以存储这个对象,我们通过postion可以了解到,所以这里使用tag可以快速定位我们以前存在缓存中的数据列表