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

Android快速开发之设计通用的ListView,GridView的适配器

2016-06-28 09:07 477 查看

概述

在Android开发的时候,经常会使用ListView或GridView来显示数据,这时就需要写对应的适配器Adapter。如果针对每个ListView或者GridView都去创建一个适配器时,那么就会出现一大堆的重复的代码。为了体现代码的简洁性,我们可以设计一个通用的适配器来避免里面重复的逻辑。

常规实现

我们先列出常规适配器的使用方法,通过分析常见的使用方式,来分析设计出通用的适配器。

1.布局文件first_activity.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" >

<ListView
android:id="@+id/listView"
android:layout_width="match_parent"
android:layout_height="match_parent" >
</ListView>

</LinearLayout>


2.Activity类中代码

package com.yuminfeng.test;

import java.util.ArrayList;
import java.util.List;
import com.yuminfeng.adapter.MyAdapter;
import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;

public class FirstActivity extends Activity{

private ListView listView;
private List<String> datas; //listview中的数据源
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.first_activity);

listView = (ListView) findViewById(R.id.listView);
datas = new ArrayList<String>();
for (int i = 0; i < 5; i++) {
datas.add("Hello "+i);
}
MyAdapter adapter = new MyAdapter(this);
adapter.setDataSource(datas);
listView.setAdapter(adapter);
}

}


3.ListView中item的布局文件item_listview.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="vertical" >

<TextView
android:id="@+id/textView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="10dp"
android:textSize="20sp" />

</LinearLayout>


4.ListView的自定义Adapter类MyAdapter.java

package com.yuminfeng.adapter;

import java.util.ArrayList;
import java.util.List;

import com.yuminfeng.test.R;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;

public class MyAdapter extends BaseAdapter{

private Context context;

private List<String> datas = new ArrayList<String>();

public MyAdapter(Context context){
this.context = context;
}

/**
* 添加数据源
* @param datas
*/
public void setDataSource(List<String> datas) {
setDataSource(datas,true);
}

/**
* 添加数据源
* @param datas
* @param isClear true:清除之前的数据源,false:追加数据源
*/
public void setDataSource(List<String> datas,boolean isClear) {
if(isClear){
this.datas.clear();
}
this.datas.addAll(datas);
notifyDataSetChanged();
}

@Override
public int getCount() {
return datas.size();
}

@Override
public Object getItem(int position) {
return datas.get(position);
}

@Override
public long getItemId(int position) {
return position;
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder viewHolder = null;
if(convertView == null){
convertView = LayoutInflater.from(context).inflate(R.layout.item_listview, parent,false);
viewHolder = new ViewHolder(convertView);
convertView.setTag(viewHolder);
}else{
viewHolder = (ViewHolder) convertView.getTag();
}
viewHolder.textView.setText(datas.get(position));
return convertView;
}

public class ViewHolder{
private TextView textView;
public ViewHolder(View convertView){
this.textView = (TextView) convertView.findViewById(R.id.textView);
}
}
}


以上就是实现ListView显示数据的基本写法,如果有10个ListView那么就会有10个这样的适配器,这10个适配器中存在大量重复的代码。为了避免这种情况,我们需要设计出一个通用的适配器,在这个通用的适配器的基础上,我们在进行不同的业务操作。

设计通用的适配器

首先分析Adapter中的包含元素:

填充布局时,需要用到Context;

需要List来存储显示的数据源;

需要布局文件来作为ListView的item;

需要ViewHolder类来优化ListView的性能;

以上几项,都是根据不同的ListView,来做不同的处理,其他重写的BaseAdapter的方法都是固定的。所以只要抽出上述的参数,灵活的进行处理,这就是设计通用的适配器的思想。

了解完了之后,我们的代码如下:

package com.yuminfeng.adapter;

import java.util.ArrayList;
import java.util.List;
import android.content.Context;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;

public abstract class CommonAdapter<E> extends BaseAdapter {

//context
private Context context;
//数据源
private List<E> mDatas = new ArrayList<E>();
//item的layout布局 id
private int layoutResId;

public CommonAdapter(Context context,int layoutResId) {
this.context = context;
this.layoutResId = layoutResId;
}

/**
* 添加数据源
* @param data
*/
public void setDataSource(List<E> data) {
setDataSource(data, true);
}

/**
* 添加数据源
* @param data
* @param isClear true:清除之前的数据源,false:追加数据源
*/
public void setDataSource(List<E> data, boolean isClear) {
if(isClear){
mDatas.clear();
}
mDatas.addAll(data);
notifyDataSetChanged();
}

/**
* 只添加一个数据
* @param data
*/
public void addData(E data){
mDatas.add(data);
notifyDataSetChanged();
}

/**
* 移除对象data的数据
* @param data
*/
public void removeData(E data){
mDatas.remove(data);
notifyDataSetChanged();
}

/**
* 通过对象移除一条数据
* @param position
*/
public void removeData(int position){
mDatas.remove(position);
notifyDataSetChanged();
}

@Override
public int getCount() {
return mDatas.size();
}

@Override
public E getItem(int position) {
return mDatas.get(position);
}

@Override
public long getItemId(int position) {
return position;
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
BaseViewHolder viewHolder = null;
if(convertView == null){
viewHolder = createViewHolder(parent,layoutResId);
if(viewHolder == null || viewHolder.getConvertView() == null){
throw new NullPointerException("createViewHolder不能返回null");
}
convertView = viewHolder.getConvertView();
convertView.setTag(viewHolder);
}else{
viewHolder = (BaseViewHolder) convertView.getTag();
}
bindViewHolder(viewHolder, position, getItem(position));

return viewHolder.getConvertView();
}

private BaseViewHolder createViewHolder(ViewGroup parent,int layoutResId){
View convertView = LayoutInflater.from(context).inflate(layoutResId, parent,false);
return new BaseViewHolder(convertView);
}

protected abstract void bindViewHolder(BaseViewHolder viewHolder,int position,E data);

public static class BaseViewHolder{
private View convertView;
private SparseArray<View> views = new SparseArray<View>();

public BaseViewHolder(View convertView){
this.convertView = convertView;
}

public View getConvertView(){
return convertView;
}

/**
* 根据View的id,取到View的对象
* @param viewId
* @return
*/
@SuppressWarnings("unchecked")
public <V> V getView(int viewId){
View view = views.get(viewId);
if(view == null){
view = convertView.findViewById(viewId);
views.put(viewId, view);
}
return (V) view;
}
}

}


上面便是设计的一个通用的适配器类,这是个抽象类。

实现时,首先继承这个类,然后重写里面的抽象方法bindViewHolder。这个方法是开放给外部的回调方法,用来设置具体的item数据。还有在创建这个子类的对象时,传入两个参数:上下文context,和布局item的id。

我们来看它的子类对象,如下:

package com.yuminfeng.adapter;

import java.util.Map;
import com.yuminfeng.test.R;
import android.content.Context;
import android.widget.ImageView;
import android.widget.TextView;

public class MyAdapter extends CommonAdapter<Map<String, Object>>{

public MyAdapter(Context context, int layoutResId) {
super(context, layoutResId);
}

@Override
protected void bindViewHolder(BaseViewHolder viewHolder,int position, Map<String, Object> data) {
TextView title = viewHolder.getView(R.id.title);
TextView content = viewHolder.getView(R.id.content);
TextView time = viewHolder.getView(R.id.time);
ImageView image = viewHolder.getView(R.id.imageView);

title.setText((String)data.get("title"));
content.setText((String)data.get("content"));
time.setText((String)data.get("time"));
image.setImageResource((Integer)data.get("icon"));
}

}


上面的类非常简单,就是给具体的item中的View设置值,其它继承CommonAdapter类的子类,也是一样的操作。

我们再看Activity中的调用,如下:

package com.yuminfeng.test;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import com.yuminfeng.adapter.MyAdapter;
import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;

public class FirstActivity extends Activity{

private ListView listView;
private List<Map<String, Object>> datas; //listview中的数据源
private int[] imageIds = new int[]{R.drawable.pic1,R.drawable.pic2,R.drawable.pic3,R.drawable.pic4,R.drawable.pic5};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.first_activity);

listView = (ListView) findViewById(R.id.listView);
datas = new ArrayList<Map<String, Object>>();

for (int i = 0; i < imageIds.length; i++) {
Map<String, Object> map = new HashMap<String, Object>();
map.put("icon",imageIds[i] );
map.put("title","美女"+i+"只");
map.put("content","这是内容,这是内容");
map.put("time","20160428");
datas.add(map);
}
MyAdapter adapter = new MyAdapter(this, R.layout.item_listview_more);
adapter.setDataSource(datas);

listView.setAdapter(adapter);
}
}


以上也是非常简单的逻辑,创建MyAdapter类,传入context和布局item。然后设置数据源后,直接使用即可。

这样就将之前重复的逻辑全部封装到抽象类CommonAdapter中,现在不需要关心了,继承该类后,只需绑定item中每个View的数据就可以了。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  android listview gridview