LeBlanc的安卓学习之旅 Adapter学习

来源:互联网 发布:装修软件手机版 编辑:程序博客网 时间:2024/06/03 20:58

LeBlanc的安卓学习之旅(一)andriod UI编程 (2)Android高级控件的使用

Adapt用法
ListView用法
1.概念
Adapter是连接后端数据和前端显示的适配器接口,是数据和UI(View)之间一个重要的纽带。在常见的View(ListView,GridView)等地方都需要用到Adapter。如下图直观的表达了Data、Adapter、View三者的关系:
这里写图片描述
Android中所有的Adapter一览:

这里写图片描述

由图可以看到在Android中与Adapter有关的所有接口、类的完整层级图。在我们使用过程中可以根据自己的需求实现接口或者继承类进行一定的扩展。比较常用的有 BaseAdapter,SimpleAdapter,ArrayAdapter,SimpleCursorAdapter等。
BaseAdapter是一个抽象类,继承它需要实现较多的方法,所以也就具有较高的灵活性;
ArrayAdapter支持泛型操作,最为简单,只能展示一行字。
SimpleAdapter有最好的扩充性,可以自定义出各种效果。
SimpleCursorAdapter可以适用于简单的纯文字型ListView,它需要Cursor的字段和UI的id对应起来。如需要实现更复杂的UI也可以重写其他方法。可以认为是SimpleAdapter对数据库的简单结合,可以方便地把数据库的内容以列表的形式展示出来。


2.应用案例
1)ArrayAdapter
列表的显示需要三个元素:
a.ListVeiw 用来展示列表的View。
b.适配器 用来把数据映射到ListView上的中介。
c.数据具体的将被映射的字符串,图片,或者基本组件。
案例一:使用ListActivity

public class Activity2 extends ListActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);               //列表项的数据        String[] Strs = {"1","2","3","4"};        //用适配器把数据映射ListView中        ArrayAdapter<String> adapter = new ArrayAdapter<String>(this,android.R.layout.simple_expandable_list_item_1,Strs);        setListAdapter(adapter);    }

案例二:

public class Activity2 extends ActionBarActivity {    private ListView listView;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        listView = new ListView(this);        listView.setAdapter(new ArrayAdapter<String>(this,android.R.layout.simple_expandable_list_item_1,getData()));        setContentView(listView);    }    public List<String> getData() {        List<String> data = new ArrayList<>();        data.add("数据1");        data.add("数据2");        data.add("数据3");        data.add("数据4");        return data;    }

上面代码使用了ArrayAdapter(Context context, int textViewResourceId, List objects)来装配数据,要装配这些数据就需要一个连接ListView视图对象和数组数据的适配器来两者的适配工作,ArrayAdapter的构造需要三个参数,依次为this,布局文件(注意这里的布局文件描述的是列表的每一行的布局,android.R.layout.simple_list_item_1是系统定义好的布局文件只显示一行文字,数据源(一个List集合)。同时用setAdapter()完成适配的最后工作。
上述是没有使用原有的Xml文件时用法,在使用原有的Xml文件时,我们要在Xml文件中添加ListView控件:

<ListView        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:id="@+id/list"></ListView>

主代码则改为

  setContentView(R.layout.activity_activity2);            listView = (ListView) findViewById(R.id.list);            listView.setAdapter(new ArrayAdapter<String>(this,android.R.layout.simple_expandable_list_item_1,getData()));            //setContentView(listView);

2)SimpleAdapter
  simpleAdapter的扩展性最好,可以定义各种各样的布局出来,可以放上ImageView(图片),还可以放上Button(按钮),CheckBox(复选框)等等。下面的代码都直接继承了ListActivity,ListActivity和普通的Activity没有太大的差别,不同就是对显示ListView做了许多优化,方面显示而已。
自定义布局:activity_simple_adapter_test.xml

<LinearLayout    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:paddingLeft="@dimen/activity_horizontal_margin"    android:paddingRight="@dimen/activity_horizontal_margin"    android:paddingTop="@dimen/activity_vertical_margin"    android:paddingBottom="@dimen/activity_vertical_margin"    android:orientation="vertical"    tools:context="com.example.admin.hello_andriod.SimpleAdatperTest">   <ImageView       android:layout_width="wrap_content"       android:layout_height="wrap_content"       android:id="@+id/img"       android:layout_margin="5dp"/>    <TextView        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:id="@+id/title"        android:textColor="#ffffff"        android:textSize="20sp"/></LinearLayout>

可以看出布局中ListView中每一项包含TextView和ImageView控件。对应在主代码中

public class SimpleAdatperTest extends ListActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        SimpleAdapter adapter = new SimpleAdapter(this,getDate(),R.layout.activity_simple_adatper_test,new String[]{"title","img"},new int[]{R.id.title,R.id.img});        setListAdapter(adapter);    }    private List<Map<String,Object>> getDate(){        List<Map<String,Object>> list = new ArrayList<Map<String,Object>>();        Map<String,Object> map = new HashMap<String, Object>();        map.put("title","摩托罗拉");        map.put("img",R.drawable.ic_launcher);        list.add(map);        map = new HashMap<String, Object>();        map.put("title","诺基亚");        map.put("img",R.drawable.ic_launcher);        list.add(map);        map = new HashMap<String, Object>();        map.put("title","三星");        map.put("img",R.drawable.ic_launcher);        list.add(map);        return list;    }

其中

 SimpleAdapter adapter = new SimpleAdapter(this,getDate(),R.layout.activity_simple_adatper_test,new String[]{"title","img"},new int[]{R.id.title,R.id.img});

函数代码中,构造函数的变量为,this,布局文件即上述我们的.xml文件,HashMap的K值为”title”,”img”,布局文件的组件id:title,img,布局文件的各组件分别映射到HashMap的各元素上,完成适配。

案例二
  下面的程序是实现一个带有图片的类表。首先需要定义好一个用来显示每一个列内容的xml,vlist.xml

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:orientation="horizontal" android:layout_width="match_parent"    android:layout_height="match_parent">    <ImageView        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:id="@+id/img"/>    <LinearLayout        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:orientation="vertical"        android:layout_margin="5px">        <TextView            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:id="@+id/title"            android:textSize="22px"/>        <TextView            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:id="@+id/ifo"            android:textSize="13px"/>        </LinearLayout></LinearLayout>

主函数代码

public class SimpleAdatperTest extends ListActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        SimpleAdapter adapter = new SimpleAdapter(this,getDate(),R.layout.vlist,new String[]{"title","img","ifo"},new int[]{R.id.title,R.id.img,R.id.ifo});        setListAdapter(adapter);    }    private List<Map<String,Object>> getDate(){        List<Map<String,Object>> list = new ArrayList<Map<String,Object>>();        Map<String,Object> map = new HashMap<String, Object>();        map.put("title","G1");        map.put("ifo","google 1");        map.put("img",R.drawable.ic_launcher);        list.add(map);        map = new HashMap<String, Object>();        map.put("title","G2");        map.put("ifo","google 2");        map.put("img",R.drawable.ic_launcher);        list.add(map);        map = new HashMap<String, Object>();        map.put("title","G3");        map.put("ifo","google3");        map.put("img",R.drawable.ic_launcher);        list.add(map);        return list;    }

效果
这里写图片描述

3)SimpleCursorAdapter
layout.activity_simple_cursor_adapter_test.xml

<LinearLayout 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:paddingLeft="@dimen/activity_horizontal_margin"    android:paddingRight="@dimen/activity_horizontal_margin"    android:paddingTop="@dimen/activity_vertical_margin"    android:paddingBottom="@dimen/activity_vertical_margin"    android:orientation="vertical"    tools:context="com.example.admin.hello_andriod.SimpleCursorAdapterTest">    <TextView        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:id="@+id/title"        android:text="SimpleCuisorAdapter测试"/>    <ListView        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:id="@+id/list">        </ListView></LinearLayout>

在ListView中加入通过adapter加入通讯录数据

public class SimpleCursorAdapterTest extends ActionBarActivity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_simple_cursor_adapter_test);        //获得一个指向系统通讯录数据库的Cursor对象获得数据来源        Cursor cur = getContentResolver().query(Contacts.People.CONTENT_URI,null,null,null,null);        startManagingCursor(cur);        ListView list = (ListView) findViewById(R.id.list);        ListAdapter adapter = new SimpleCursorAdapter(this,android.R.layout.simple_list_item_1,cur,new String[]{Contacts.People.NAME},new int[]{android.R.id.text1});        list.setAdapter(adapter);    }

一定要以数据库作为数据源的时候,才能使用SimpleCursorAdapter,这里特别需要注意的一点是:不要忘了在AndroidManifest.xml文件中加入权限

<uses-permission android:name="android.permission.READ_CONTACTS"></uses-permission>

效果如下
这里写图片描述

4)BaseAdapter
有时候,列表不光会用来做显示用,我们同样可以在在上面添加按钮。添加按钮首先要写一个有按钮的xml文件,然后自然会想到用上面的方法定义一个适配器,然后将数据映射到布局文件上。但是事实并非这样,因为按钮是无法映射的,即使你成功的用布局文件显示出了按钮也无法添加按钮的响应,这时就要研究一下ListView是如何现实的了,而且必须要重写一个类继承BaseAdapter。下面的示例将显示一个按钮和一个图片,两行字如果单击按钮将删除此按钮的所在行。并告诉你ListView究竟是如何工作的。
带有Button控件的布局文件vlist2

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:orientation="horizontal" android:layout_width="match_parent"    android:layout_height="match_parent">    <ImageView        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:id="@+id/img"        android:layout_margin="5px"/>    <LinearLayout        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:orientation="vertical">        <TextView            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:id="@+id/title"            android:textSize="22px"/>        <TextView            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:id="@+id/info"            android:textSize="13px"/>        </LinearLayout>    <Button        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:id="@+id/btnView"        android:text="s_view_btn"        android:layout_gravity="bottom|right"/></LinearLayout>

主函数代码

public class BaseAdaptertest extends ListActivity {    private List<Map<String,Object>> mData;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        mData = getData();        MyAdapter adapter = new MyAdapter(this);        setListAdapter(adapter);    }    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.ic_launcher);                    list.add(map);                    map = new HashMap<String, Object>();                    map.put("title", "G2");                    map.put("info", "google 2");                    map.put("img", R.drawable.ic_launcher);                    list.add(map);                    map = new HashMap<String, Object>();                    map.put("title", "G3");                    map.put("info", "google 3");                    map.put("img", R.drawable.ic_launcher);                    list.add(map);                    return list;                }    public class MyAdapter extends BaseAdapter {        private LayoutInflater mImflater;        //不同点是LayoutInflater是用来找res/layout/下的xml布局文件,并且实例化;而findViewById()是找xml布局文件下的具体widget控件(如Button、TextView等)。        //构造函数        public MyAdapter(Context context)        {            this.mImflater = LayoutInflater.from(context);        }        @Override        public int getCount() {            return mData.size();        }        @Override        public Object getItem(int position) {            return null;        }        @Override        public long getItemId(int position) {            return 0;        }        @Override        public View getView(int position, View convertView, ViewGroup parent) {            ViewHolder holder = null;            if(convertView == null)            {                holder = new ViewHolder();                convertView = mImflater.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.btnView);            }            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(new View.OnClickListener() {                  @Override                  public void onClick(View v) {                    showInfo();                   }                 });            return convertView;        }    }    public void showInfo(){        new AlertDialog.Builder(this).setTitle("我的ListView").setMessage("介绍...").setPositiveButton("确定",new DialogInterface.OnClickListener(){            @Override            public void onClick(DialogInterface dialog, int which) {            }        }).show();    }    public final class ViewHolder{        public ImageView img;        public TextView title;        public TextView info;        public Button viewBtn;    }}

下面将对上述代码,做详细的解释,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了。

结果如下
这里写图片描述
这里写图片描述

0 0