各种Adapter的用法

转载 2012年03月27日 21:26:18
摘要:   同样是一个ListView,可以用不同的Adapter让它显示出来,比如说最常用的ArrayAdapter,SimpleAdapter,SimpleCursorAdapter,以及重写BaseAdapter等方法。  ArrayAdapter比较简单,但它只能用于显示文字。而Si ...
同样是一个ListView,可以用不同的Adapter让它显示出来,比如说最常用的ArrayAdapter,SimpleAdapter,SimpleCursorAdapter,以及重写BaseAdapter等方法。
  ArrayAdapter比较简单,但它只能用于显示文字。而SimpleAdapter则有很强的扩展性,可以自定义出各种效果,SimpleCursorAdapter则可以从数据库中读取数据显示在列表上,通过从写BaseAdapter可以在列表上加处理的事件等。
  下面先来看看ArrayAdapter:
  1 package com.shang.test;
  2
  3  import java.util.ArrayList;
  4
  5  import android.app.Activity;
  6  import android.os.Bundle;
  7  import android.widget.ArrayAdapter;
  8  import android.widget.ListView;
  9
  10  /**
  11  *
  12  * @author shangzhenxiang
  13  *
  14  */
  15  public class TestArrayAdapterActivity extends Activity{
  16
  17     private ListView mListView;
  18     private ArrayList<String> mArrayList = new ArrayList<String>();
  19
  20     @Override
  21     protected void onCreate(Bundle savedInstanceState) {
  22         super.onCreate(savedInstanceState);
  23         setContentView(R.layout.testarrayadapter);
  24         mListView = (ListView) findViewById(R.id.myArrayList);
  25         mListView.setAdapter(new ArrayAdapter<String>(this, android.R.layout.simple_expandable_list_item_1, getData()));
  26     }
  27
  28     private ArrayList<String> getData() {
  29         mArrayList.add("测试数据1");
  30         mArrayList.add("测试数据2");
  31         mArrayList.add("测试数据3");
  32         mArrayList.add("测试数据4");
  33         mArrayList.add("测试数据5");
  34         mArrayList.add("测试数据6");
  35         return mArrayList;
  36     }
  37 }
  布局里面有个ListView就可以了:
  1 <?xml version="1.0" encoding="utf-8"?>
  2  <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3     android:orientation="vertical"
  4     android:layout_width="match_parent"
  5     android:layout_height="fill_parent">
  6     <TextView
  7         android:layout_width="match_parent"
  8         android:layout_height="wrap_content"
  9         android:text="@string/hello"/>
  10     <ListView
  11         android:id="@+id/myArrayList"
  12         android:layout_width="match_parent"
  13         android:layout_height="wrap_content"/>
  14  </LinearLayout>
  上面的代码中用到了ArrayAdapter的构造方法:
  public ArrayAdapter (Context context, int textViewResourceId, T[] objects)


  Api中是这么描述的:
  其中Context为当前的环境变量,可以显示一行文字的一个布局文件,和一个List的集合,也就是数据源。
  布局文件可以自己写,也可以用系统的,我这里是用的系统的。自己写的布局中包含一个TextView就可以了,当然系统中也有包含一个TextView的布局文件:就是 android.R.layout.simple_expandable_list_item_1,调用这个比较方便。
  这里给出运行后的效果图:


  下面说说SimpleCursorAdapter:
  Api中是这么说的:An easy adapter to map columns from a cursor to TextViews or ImageViews defined in an XML file. You can specify which columns you want, which views you want to display the columns, and the XML file that defines the appearance of these views.
  简单的说就是 方便把Cursor中得到的数据进行列表显示,并可以把指定的列映射到指定的TextView上。
  我这里写的是从联系人中拿到数据并显示在列表上。代码如下:
  1 package com.shang.test;
  2
  3  import android.app.Activity;
  4  import android.database.Cursor;
  5  import android.os.Bundle;
  6  import android.provider.Contacts.People;
  7  import android.widget.ListView;
  8  import android.widget.SimpleCursorAdapter;
  9
  10  /**
  11  *
  12  * @author shangzhenxiang
  13  *
  14  */
  15 public class TestSimpleCursorAdapter extends Activity {
  16
  17     private ListView mListView;
  18     private Cursor mCursor;
  19     private SimpleCursorAdapter mAdapter;
  20
  21     @Override
  22     protected void onCreate(Bundle savedInstanceState) {
  23         super.onCreate(savedInstanceState);
  24         setContentView(R.layout.testsimplecursoradapter);
  25         mListView = (ListView) findViewById(R.id.mySimpleCursorList);
  26         mCursor = getContentResolver().query(People.CONTENT_URI, null, null, null, null);
  27         startManagingCursor(mCursor);
  28         mAdapter = new SimpleCursorAdapter(TestSimpleCursorAdapter.this, android.R.layout.simple_expandable_list_item_1, mCursor, new String[]{People.NAME}, new int[]{android.R.id.text1});
  29         mListView.setAdapter(mAdapter);
  30     }
  31 }
  mCursor = getContentResolver().query(People.CONTENT_URI, null, null, null, null);是先获得一个指向系统联系人的Cursor
  startManagingCursor(mCursor);是指我们把Cursor交给这个Activity保管,这样Cursor便会和Activity同步,我们不用手动管理了。
  simpleCursorAdapter API中是这样说的:


  其中前面的2个参数跟ArrayAdapter中是一样的,第三个参数是传个来的参数, 其实也是数据源,后面的2个参数是2个数组,前一个是String【】类型的,而后一个是int【】类型的,说明前一个参数中的值对应的是从数据库中的字段,后一个是布局文件中和这个字段对应的id,也就是说这个字段对应得值要显示在哪里(比如说我们这里查到的联系人中的NAME字段,要显示在一个对应的TextView上面)。
  这里我们具体看一下系统的布局,也就是我们这里的第二个参数的布局,便于理解,android.R.layout.simple_expandable_list_item_1.xml文件中是这么写的:
  1 <?xml version="1.0" encoding="utf-8"?>
  2 <!-- Copyright (C) 2006 The Android Open Source Project
  3
  4      Licensed under the Apache License, Version 2.0 (the "License");
  5      you may not use this file except in compliance with the License.
  6      You may obtain a copy of the License at
  7
  8           http://www.apache.org/licenses/LICENSE-2.0
  9
  10      Unless required by applicable law or agreed to in writing, software
  11      distributed under the License is distributed on an "AS IS" BASIS,
  12      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13      See the License for the specific language governing permissions and
  14      limitations under the License.
  15 -->
  16
  17 <TextView xmlns:android="http://schemas.android.com/apk/res/android"
  18     android:id="@android:id/text1"
  19     android:layout_width="match_parent"
  20     android:layout_height="?android:attr/listPreferredItemHeight"
  21     android:paddingLeft="?android:attr/expandableListPreferredItemPaddingLeft"
  22     android:textAppearance="?android:attr/textAppearanceLarge"
  23     android:gravity="center_vertical"
  24 />
  注意他有一个id,这个id也是系统的id,这个布局中只有一个TextView,所以只能显示一个字段,我们这里显示的联系人的名字,
  而最后的一个参数就是由这么写id组成的一个数据(如果有很多TextView的话)。比如说我们要显示很多字段,布局文件中就要写很多TextView,而每一个TextView都有一个ID,第三个参数中有多少个字段,第四个参数中就有多少个id,并一一对应。
  我们来看一下运行效果图:


  上面说到的2种方法都是显示的文字,比方说我们要显示图片怎么办呢,还要显示很多内容,还要按自己喜欢的布局排列怎么办呢,用SimpleAdapter,扩展性好,可以定义各种各样的布局。
  代码如下:
  1 package com.shang.test;
  2
  3 import java.util.ArrayList;
  4 import java.util.HashMap;
  5 import java.util.List;
  6
  7 import android.app.Activity;
  8 import android.os.Bundle;
  9 import android.widget.ListView;
  10 import android.widget.SimpleAdapter;
  11
  12 /**
  13  *
  14  * @author shangzhenxiang
  15  *
  16  */
  17 public class TestSimpleAdapter extends Activity {
  18
  19     private ListView mListView;
  20     private SimpleAdapter mAdapter;
  21     private List<HashMap<String, Object>> mHashMaps;
  22     private HashMap<String, Object> map;
  23
  24     @Override
  25     protected void onCreate(Bundle savedInstanceState) {
  26         super.onCreate(savedInstanceState);
  27         setContentView(R.layout.testsimpleadapter);
  28         mListView = (ListView) findViewById(R.id.mySimpleList);
  29         mAdapter = new SimpleAdapter(this, getData(), R.layout.simpleitem, new String[]{"image", "title", "info"}, new int[]{R.id.img, R.id.title, R.id.info});
  30         mListView.setAdapter(mAdapter);
  31     }
  32
  33     private List<HashMap<String, Object>> getData() {
  34         mHashMaps = new ArrayList<HashMap<String,Object>>();
  35         map = new HashMap<String, Object>();
  36         map.put("image", R.drawable.gallery_photo_1);
  37         map.put("title", "G1");
  38         map.put("info", "google 1");
  39         mHashMaps.add(map);
  40
  41         map = new HashMap<String, Object>();
  42         map.put("image", R.drawable.gallery_photo_2);
  43         map.put("title", "G2");
  44         map.put("info", "google 2");
  45         mHashMaps.add(map);
  46
  47         map = new HashMap<String, Object>();
  48         map.put("image", R.drawable.gallery_photo_3);
  49         map.put("title", "G3");
  50         map.put("info", "google 3");
  51
  52         mHashMaps.add(map);
  53         return mHashMaps;
  54     }
  55 }
  simpleAdapter的数据都是用HashMap构成的List,List里面的每一节对应的是ListView的没一行,这里先建一个HashMap构成的List,布局中有3个元素,ImageView,2个TextView,每个item项的布局文件如下:
  1 <?xml version="1.0" encoding="utf-8"?>
  2 <LinearLayout
  3   xmlns:android="http://schemas.android.com/apk/res/android"
  4   android:layout_width="match_parent"
  5   android:layout_height="match_parent"
  6   android:orientation="horizontal">
  7     <ImageView
  8         android:layout_width="wrap_content"
  9         android:id="@+id/img"
  10         android:layout_margin="5px"
  11         android:layout_height="wrap_content">
  12         </ImageView>
  13     <LinearLayout
  14         android:id="@+id/linearLayout1"
  15         android:layout_width="wrap_content"
  16         android:layout_height="wrap_content"
  17         android:orientation="vertical">
  18         <TextView
  19             android:id="@+id/title"
  20             android:layout_width="wrap_content"
  21             android:layout_height="wrap_content"
  22             android:textColor="#ffffff"
  23             android:textSize="22px"></TextView>
  24         <TextView
  25             android:id="@+id/info"
  26             android:layout_width="wrap_content"
  27             android:layout_height="wrap_content"
  28             android:textColor="#ffffff"
  29             android:textSize="13px"></TextView>
  30     </LinearLayout>
  31 </LinearLayout>
  所以有了HashMap构成的数组后,我们要在HashMap中加入数据,按顺序加入图片,title,info,一个HashMap就构成了ListView中的一个Item项,我们在看下API中是怎么描述simpleAdapter的:


  第一个参数和第三个参数跟ArrayAdapter中的是一样的,第二个参数就是由HashMap组成的List,也就是数据源,而第5个参数也就是map中的key,最后一个参数就是map中key对应的值要显示在布局中的位置的id。
  看下效果:


  如果我们想在每个Item中加个button,而且点击button有对应的操作,那该怎么办呢。
  这时我们可以重写baseAdapter,看代码:
  1 package com.shang.test;
  2
  3 import java.util.ArrayList;
  4 import java.util.HashMap;
  5 import java.util.List;
  6
  7 import android.app.Activity;
  8 import android.app.AlertDialog;
  9 import android.content.Context;
  10 import android.content.DialogInterface;
  11 import android.os.Bundle;
  12 import android.view.LayoutInflater;
  13 import android.view.View;
  14 import android.view.View.OnClickListener;
  15 import android.view.ViewGroup;
  16 import android.widget.BaseAdapter;
  17 import android.widget.Button;
  18 import android.widget.ImageView;
  19 import android.widget.ListView;
  20 import android.widget.TextView;
  21
  22 /**
  23  *
  24  * @author shangzhenxiang
  25  *
  26  */
  27 public class TestBaseAdapter extends Activity {
  28
  29     private ListView mListView;
  30
  31     @Override
  32     protected void onCreate(Bundle savedInstanceState) {
  33         super.onCreate(savedInstanceState);
  34         setContentView(R.layout.baseadapterlist);
  35         mListView = (ListView) findViewById(R.id.baselist);
  36         mListView.setAdapter(new BaseListAdapter(this));
  37     }
  38
  39     private List<HashMap<String, Object>> getData() {
  40         List<HashMap<String, Object>> maps = new ArrayList<HashMap<String,Object>>();
  41         HashMap<String, Object> map = new HashMap<String, Object>();
  42         map.put("image", R.drawable.gallery_photo_1);
  43         map.put("title", "G1");
  44         map.put("info", "google 1");
  45         maps.add(map);
  46
  47         map = new HashMap<String, Object>();
  48         map.put("image", R.drawable.gallery_photo_2);
  49         map.put("title", "G2");
  50         map.put("info", "google 2");
  51         maps.add(map);
  52
  53         map = new HashMap<String, Object>();
  54         map.put("image", R.drawable.gallery_photo_3);
  55         map.put("title", "G3");
  56         map.put("info", "google 3");
  57         maps.add(map);
  58         return maps;
  59     }
  60
  61     private class BaseListAdapter extends BaseAdapter implements OnClickListener {
  62
  63         private Context mContext;
  64         private LayoutInflater inflater;
  65
  66         public BaseListAdapter(Context mContext) {
  67             this.mContext = mContext;
  68             inflater = LayoutInflater.from(mContext);
  69         }
  70
  71         @Override
  72         public int getCount() {
  73             return getData().size();
  74         }
  75
  76         @Override
  77         public Object getItem(int position) {
  78             return null;
  79         }
  80
  81         @Override
  82         public long getItemId(int position) {
  83             return 0;
  84         }
  85
  86         @Override
  87         public View getView(int position, View convertView, ViewGroup parent) {
  88             ViewHolder viewHolder = null;
  89             if(convertView == null) {
  90                 viewHolder = new ViewHolder();
  91                 convertView = inflater.inflate(R.layout.testbaseadapter, null);
  92                 viewHolder.img = (ImageView) convertView.findViewById(R.id.img);
  93                 viewHolder.title = (TextView) convertView.findViewById(R.id.title);
  94                 viewHolder.info = (TextView) convertView.findViewById(R.id.info);
  95                 viewHolder.button = (Button) convertView.findViewById(R.id.basebutton);
  96                 convertView.setTag(viewHolder);
  97             } else {
  98                 viewHolder = (ViewHolder) convertView.getTag();
  99             }
  100
  101             System.out.println("viewHolder = " + viewHolder);
  102             viewHolder.img.setBackgroundResource((Integer) getData().get(position).get("image"));
  103             viewHolder.title.setText((CharSequence) getData().get(position).get("title"));
  104             viewHolder.info.setText((CharSequence) getData().get(position).get("info"));
  105             viewHolder.button.setOnClickListener(this);
  106
  107             return convertView;
  108         }
  109
  110         class ViewHolder {
  111             ImageView img;
  112             TextView title;
  113             TextView info;
  114             Button button;
  115         }
  116
  117         @Override
  118         public void onClick(View v) {
  119             int id = v.getId();
  120             switch(id) {
  121             case R.id.basebutton:
  122                 showInfo();
  123                 break;
  124             }
  125         }
  126
  127         private void showInfo() {
  128             new AlertDialog.Builder(TestBaseAdapter.this).setTitle("my listview").setMessage("introduce....").
  129             setPositiveButton("OK", new DialogInterface.OnClickListener() {
  130
  131                 @Override
  132                 public void onClick(DialogInterface dialog, int which) {
  133                     // TODO Auto-generated method stub
  134
  135                 }
  136             }).show();
  137         }
  138     }
  139 }
  在看下item的布局文件:
  1 <?xml version="1.0" encoding="utf-8"?>
  2 <LinearLayout
  3   xmlns:android="http://schemas.android.com/apk/res/android"
  4   android:layout_width="match_parent"
  5   android:layout_height="match_parent"
  6   android:orientation="horizontal">
  7     <ImageView
  8         android:layout_width="wrap_content"
  9         android:id="@+id/img"
  10         android:layout_margin="5px"
  11         android:layout_height="wrap_content">
  12         </ImageView>
  13     <LinearLayout
  14         android:id="@+id/linearLayout1"
  15         android:layout_width="wrap_content"
  16         android:layout_height="wrap_content"
  17         android:orientation="vertical">
  18         <TextView
  19             android:id="@+id/title"
  20             android:layout_width="wrap_content"
  21             android:layout_height="wrap_content"
  22             android:textColor="#ffffff"
  23             android:textSize="22px"></TextView>
  24         <TextView
  25             android:id="@+id/info"
  26             android:layout_width="wrap_content"
  27             android:layout_height="wrap_content"
  28             android:textColor="#ffffff"
  29             android:textSize="13px"></TextView>
  30     </LinearLayout>
  31
  32     <Button
  33         android:id="@+id/basebutton"
  34         android:text="more"
  35         android:focusable="false"
  36         android:layout_gravity="bottom|right"
  37         android:layout_height="wrap_content"
  38         android:layout_width="wrap_content"/>
  39 </LinearLayout>
  listView在开始绘制的时候,系统首先调用getCount()函数,根据他的返回值得到listView的长度(这也是为什么在开始的第一张图特别的标出列表长度),然后根据这个长度,调用getView()逐一绘制每一行。如果你的getCount()返回值是0的话,列表将不显示同样 return 1,就只显示一行。
  如果我们要自定义适配器,那就要重写getView方法,getView()有三个参数,position表示将显示的是第几行,covertView是从布局文件中inflate来的布局。我们写一个类来描述布局文件中的各个组件,比如ImageView,TextView等,然后判断convertView是否为空,如果为空就从inflate中拿到布局,并新建一个ViewHolder,然后从convertView中拿到布局中的各个组件,同时把ViewHolder放到tag中去,下次就不用重写new了,直接从tag中拿就可以了,然后把布局中的各个组件都设上对应的值,这里的Position对应到含有HashMap的List中的position。
  在实际的运行过程中会发现listView的每一行没有焦点了,这是因为Button抢夺了listView的焦点,只要布局文件中将Button设置为没有焦点就OK了。
  看下运行效果:

Java 实现适配器(Adapter)模式

平时我们会经常碰到这样的情况,有了两个现成的类,它们之间没有什么联系,但是我们现在既想用其中一个类的方法,同时也想用另外一个类的方法。有一个解决方法是,修改它们各自的接口,但是这是我们最不愿意看到的。...
  • jjwwmlp456
  • jjwwmlp456
  • 2014年10月08日 15:21
  • 7899

注解式框架ButterKnife的用法及在adapter中使用的优化

注解式框架的使用将会大大简化代码编写量,提升开发效率,主流的注解式框架有Dagger,ButterKnife,AndrodAnnotations。AndrodAnnotations配置麻烦,需要在项目...
  • zpc5925439
  • zpc5925439
  • 2015年06月26日 15:09
  • 4409

Java设计模式透析之 —— 适配器(Adapter)

今天一大早,你的leader就匆匆忙忙跑过来找到你:“快,快,紧急任务!最近ChinaJoy马上就要开始了,老板要求提供一种直观的方式,可以查看到我们新上线的游戏中每个服的在线人数。” 你看了看日期,...
  • sinyu890807
  • sinyu890807
  • 2013年07月25日 09:01
  • 39841

Java设计模式之适配器模式(Adapter Pattern)

Adapter Pattern的作用是在不改变功能的前提下转换接口。Adapter分为两类,一类是Object Adapter, 另一类是Class Adapter。由于Class Adapter的实...
  • tracker_w
  • tracker_w
  • 2014年06月17日 03:39
  • 2402

适配ListView的几种常见Adapter的用法总结

万事开头难
  • jinxin_cuit
  • jinxin_cuit
  • 2016年09月07日 10:02
  • 1252

Java 实现适配器(Adapter)模式

平时我们会经常碰到这样的情况,有了两个现成的类,它们之间没有什么联系,但是我们现在既想用其中一个类的方法,同时也想用另外一个类的方法。有一个解决方法是,修改它们各自的接口,但是这是我们最不愿意看到的。...
  • jjwwmlp456
  • jjwwmlp456
  • 2014年10月08日 15:21
  • 7899

Android 中的adapter和作用以及常见的adapter

简述: adapter是数据与ui之间的桥梁,它把后台数据与前端ui连接到一起,是一个展示数据的载体。 常用的adapter: BaseAdapter : 基础适配器,对于spinner listvi...
  • zl18603543572
  • zl18603543572
  • 2015年11月21日 23:38
  • 4655

关于Adapter的notifyDataSetChanged无法刷新的问题。

1、notifyDataSetChanged()方法必须放到UI线程中。 2、adapter中的数据源改变前和改变后引用的对象必须是同一个。 修改前(无效): import...
  • cqrf2006
  • cqrf2006
  • 2015年04月29日 14:31
  • 1165

Android Adapter嵌套Adapter(文档类app,说明书类app)

相信很多人都用过Adapter,但是Adapter嵌套Adapter确很少使用,主要是这样的需求也不多,最近要做一个说明书类的应用,左边实现目录的嵌套,点击一级目录弹出二级目录,一级目录为一个Adap...
  • yuxuehandong
  • yuxuehandong
  • 2016年06月09日 21:05
  • 1934

Java设计模式之适配器模式(Adapter)在Android中的应用

1.概念:将内容与控件相分离的一种设计(公式) ,达到内容灵活显示的目的。2.适配器模式主要分为三类:类的适配器模式、对象的适配器模式、接口的适配器模式。单独讲解接口的适配器模式a. 接口的适配器模式...
  • huang_xiao_yu
  • huang_xiao_yu
  • 2015年11月01日 00:33
  • 1098
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:各种Adapter的用法
举报原因:
原因补充:

(最多只允许输入30个字)