您的位置:首页 > 移动开发 > Android开发

Android 02:Android中Adapter用法总结

2015-08-03 23:20 721 查看
1. 概念

Adapter是连接后端数据和前端显示的适配器接口,是数据和UI(View)之间一个重要的纽带。在常见的View(List View,Grid View)等地方都需要用到Adapter。如下图直观的表达了Data、Adapter、View三者的关系:



Android中所有的Adapter一览:



由图可以看到在Android中与Adapter有关的所有接口、类的完整层级图。在我们使用过程中可以根据自己的需求实现接口或者继承类进行一定的扩展。下面是一些常用的:

*ArrayAdapter支持泛型操作,最为简单,只能展示一行字。

*SimpleAdapter有最好的扩充性,可以自定义出各种效果。

*BaseAdapter是一个抽象类,继承它需要实现较多的方法,所以也就具有较高的灵活性。

*SimpleCursorAdapter可以适用于简单的纯文字ListView,它需要Cursor的字段和UI的id对应起来。如需要实现更复杂的UI也可以重写其他方法。可以认为是SimpleAdapter对数据库的简单结合,可以方便地把数据库的内容以列表的形式展示出来。

2. 应用案例

1)ArrayAdapter

列表的显示需要三个元素:

a.ListVeiw 用来展示列表的View。

b.适配器 用来把数据映射到ListView上的中介。

c.数据 具体的将被映射的字符串,图片,或者基本组件。

经典案例

public class MainActivity  extends Activity{

private ListView mListView;
private ArrayList<String> mArrayList = new ArrayList<String>();

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.testarrayadapter);
//获取布局中的listview
mListView = (ListView) findViewById(R.id.myList);
//添加适配器
mListView.setAdapter(new ArrayAdapter<String>(this, android.R.layout.simple_expandable_list_item_1, getData()));
}
//Listview中的数据
private ArrayList<String> getData() {
mArrayList.add("测试数据1");
mArrayList.add("测试数据2");
mArrayList.add("测试数据3");
mArrayList.add("测试数据4");
mArrayList.add("测试数据5");
mArrayList.add("测试数据6");
return mArrayList;
}
}


上面的代码中用到了ArrayAdapter的构造方法:

public ArrayAdapter (Context context, int textViewResourceId, T[] objects)

API中的描述



该构造方法用来装配数据,要装配这些数据就需要一个连接ListView视图对象和数组数据的适配器来完成两者的适配工作ArrayAdapter的构造需要三个参数,依次为

this,

布局文件(注意这里的布局文件描述的是列表的每一行的布局,android.R.layout.simple_list_item_1是系统定义好的布局文件只显示一行文字,

数据源(一个List集合)

同时用setAdapter()完成适配的最后工作

布局文件activity_main:加一个Listview就行了:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="fill_parent">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/hello"/>
<ListView
android:id="@+id/myList"
android:layout_width="match_parent"
android:layout_height="wrap_content"/>
</LinearLayout>


效果图如下:



2)SimpleAdapter

simpleAdapter的扩展性最好,可以定义各种各样的布局出来,可以放上ImageView(图片),还可以放上Button(按钮),CheckBox(复选框)

经典案例

public class MainActivity extends Activity {

private ListView mListView;
private SimpleAdapter mAdapter;
private List<HashMap<String, Object>> mHashMaps;
private HashMap<String, Object> map;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mListView = (ListView) findViewById(R.id.myList);
mAdapter = new SimpleAdapter(this, getData(), R.layout.simple, new String[]{"image", "title", "info"}, new int[]{R.id.img, R.id.title, R.id.info});
mListView.setAdapter(mAdapter);
}

private List<HashMap<String, Object>> getData() {
mHashMaps = new ArrayList<HashMap<String,Object>>();
map = new HashMap<String, Object>();
map.put("image", R.drawable.ic_launcher);
map.put("title", "标题1");
map.put("info", "android 1");
mHashMaps.add(map);

map = new HashMap<String, Object>();
map.put("image", R.drawable.ic_launcher);
map.put("title", "标题2");
map.put("info", "android 2");
mHashMaps.add(map);

map = new HashMap<String, Object>();
map.put("image", R.drawable.ic_launcher);
map.put("title", "标题3");
map.put("info", "android 3");

mHashMaps.add(map);
return mHashMaps;
}
}


上述代码中用到simpleAdapter,我们在看下API中是怎么描述:



第一个参数和第三个参数跟ArrayAdapter中的是一样的

第二个是数据源:保存有每一行数据的Map构成的HashMap对象,也就是说,每一行数据里的每一个属性由它的名字值构成一个键值对,多个键值对构成这一行的一个Map对象,再由多行的多个Map对象够成一个HashMap对象。

第四个String数组是键的集合,第五个int数组是布局文件中控件ID的集合,也就是说,从String数组里的键名去找到值,然后将这个值赋给在int数组里位置对应的控件。

布局文件有两个:

主布局文件:activity_main同上

simple.xml : simpleAdapter的数据都是用HashMap构成的List,List里面的每一节对应的是ListView的每一行,布局中有3个元素,ImageView,2个TextView,每个item项的布局文件如下:

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


代码效果:



3)baseAdapter

在上面的例子中添加按钮,并对按钮进行操作,simpleadapter可以添加,但是无法获取焦点。这时就要重写baseAdapter

使用BaseAdapter必须写一个类继承它,同时BaseAdapter是一个抽象类,继承它必须实现它的方法

public class MainActivity extends Activity {

private ListView mListView;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);

mListView = (ListView) findViewById(R.id.myList);
mListView.setAdapter(new BaseListAdapter(this));
}

private List<HashMap<String, Object>> getData() {
List<HashMap<String, Object>> maps = new ArrayList<HashMap<String,Object>>();
HashMap<String, Object> map = new HashMap<String, Object>();
map.put("image", R.drawable.ic_launcher);
map.put("title", "标题1");
map.put("info", "android sdk 1");
maps.add(map);

map = new HashMap<String, Object>();
map.put("image", R.drawable.ic_launcher);
map.put("title", "标题2");
map.put("info", "android sdk 2");
maps.add(map);

map = new HashMap<String, Object>();
map.put("image", R.drawable.ic_launcher);
map.put("title", "标题3");
map.put("info", "android sdk 3");
maps.add(map);
return maps;
}
//新建一个类继承BaseAdapter,实现视图与数据的绑定
private class BaseListAdapter extends BaseAdapter implements OnClickListener {

private Context mContext;
//得到一个LayoutInfalter对象用来导入布局
private LayoutInflater inflater;

// 四个构造函数
public BaseListAdapter(Context mContext) {
this.mContext = mContext;
inflater = LayoutInflater.from(mContext);
}

@Override
//getCount的值是程序加载显示到UI之前读取的,它获得的值决定listview共显示多少行
public int getCount() {
return getData().size();
}

@Override
//根据ListView所在的位置返回view
public Object getItem(int position) {
return null;
}

@Override
//根据ListView位置返回数据源id
public long getItemId(int position) {
return 0;
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder viewHolder = null;
if(convertView == null) {
viewHolder = new ViewHolder();
//把布局变成view(layoutinflate的用法)
convertView = inflater.inflate(R.layout.simple, null);
//得到各个控件的对象
viewHolder.img = (ImageView) convertView.findViewById(R.id.img);
viewHolder.title = (TextView) convertView.findViewById(R.id.title);
viewHolder.info = (TextView) convertView.findViewById(R.id.info);
viewHolder.button = (Button) convertView.findViewById(R.id.basebutton);
//绑定ViewHolder对象
convertView.setTag(viewHolder);
} else {
//取出ViewHolder对象
viewHolder = (ViewHolder) convertView.getTag();
}
//设置每个item显示的内容,即我们存放在动态数组中的数据.取出map根据键值对取值
viewHolder.img.setBackgroundResource((Integer) getData().get(position).get("image"));
viewHolder.title.setText((CharSequence) getData().get(position).get("title"));
viewHolder.info.setText((CharSequence) getData().get(position).get("info"));
viewHolder.button.setOnClickListener(this);

return convertView;
}

//存放控件
class ViewHolder {
ImageView img;
TextView title;
TextView info;
Button button;
}

@Override
public void onClick(View v) {
int id = v.getId();
switch(id) {
case R.id.basebutton:
showInfo();
break;
}
}

private void showInfo() {
new AlertDialog.Builder(MainActivity.this).setTitle("my listview").setMessage("baseadapter....").
setPositiveButton("OK", new DialogInterface.OnClickListener() {

@Override
public void onClick(DialogInterface dialog, int which) {
// TODO Auto-generated method stub

}
}).show();
}

}
}


代码解释:如果我们要自定义适配器,那就要重写getView方法,getView()有三个参数,position表示将显示的是第几 行,covertView是从布局文件中inflate来的布局。

我们写一个类来描述布局文件中的各个组件,比如ImageView,TextView 等,然后判断convertView是否为空,如果为空就从inflate中拿到布局,并新建一个ViewHolder,然后从convertView中 拿到布局中的各个组件,同时把ViewHolder放到tag中去,下次就不用重写new了,直接从tag中拿就可以了,然后把布局中的各个组件都设上对 应的值,这里的Position对应到含有HashMap的List中的position。

布局文件有两个:

主布局文件:activity_main同上

simple.xml :

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal">
<ImageView
android:layout_width="wrap_content"
android:id="@+id/img"
android:layout_margin="5px"
android:layout_height="wrap_content">
</ImageView>
<LinearLayout
android:id="@+id/linearLayout1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
//保证两个文本框竖着放
android:orientation="vertical">
<TextView
android:id="@+id/title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#ffffff"
android:textSize="22px"></TextView>
<TextView
android:id="@+id/info"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#ffffff"
android:textSize="13px"></TextView>
</LinearLayout>

<Button
android:id="@+id/basebutton"
android:text="more"
**android:focusable="false"**
android:layout_gravity="bottom|right"
android:layout_height="wrap_content"
android:layout_width="wrap_content"/>
</LinearLayout>


需要注意的是,Button会抢夺ListView的焦点,需要将Button设置为没有焦点。设置非常简单,只需要在xml的Button标签下加入一行:android:focusable=“false”代码就可以了。

效果:



4)SimpleCursorAdapter

简单的说就是 方便把Cursor中得到的数据进行列表显示,并可以把指定的列映射到ListView的每个item上

下面是一个获取手机联系人数据的例子:

public class MainActivity extends Activity {

private ListView mListView;
private Cursor mCursor;
private SimpleCursorAdapter mAdapter;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);

mListView = (ListView) findViewById(R.id.myList);
//获得一个指向系统通讯录数据库的Cursor对象获得数据来源
mCursor = getContentResolver().query(People.CONTENT_URI, null, null, null, null);
//把Cursor交给本Activity保管,使得Cursor与Activity同步,这样就不用手动管理了。
startManagingCursor(mCursor);
//实例化列表适配器
mAdapter = new SimpleCursorAdapter(MainActivity.this, android.R.layout.simple_expandable_list_item_1, mCursor, new String[]{People.NAME}, new int[]{android.R.id.text1});
mListView.setAdapter(mAdapter);
}
}


SimpleCursorAdapter在API中描述



第一个和第二个个参数跟ArrayAdapter中是一样的

第三个参数是传个来的参数,是数据源,

第四个是String【】类型的,第五个是int【】类型的,前一个参数中的值对应的是从数据库中的字 段,后一个是布局文件中和这个字段对应的id,也就是说这个字段对应得值要显示在哪里。

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

<uses-permission android:name="android.permission.READ_CONTACTS"></uses-permission>
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: