Android中万能的适配器的详细讲解(附源代码)

Android中万能的适配器的详细讲解
在Android开发中,适配器的用处是非常大的,尤其是效率优化方面。除了使用ViewHolder复用View之外,如果存在很多的ListView或者是一个ListView中存在很多的View组件,那对代码的阅读不是很好的。考虑到优化以及共通方面,我封装了ViewHolder类以及将Adapter类封装成共通的了,将对以后的开发带来很大的方便。
适配器中提高性能优化如下:
1.利用convertView

利用Android的Recycler机制,利用convertView来重新回收View,效率有了本质提高。View的每次创建是比较耗时的,因此对于getview方法传入的convertView应充分利用 != null的判断 。

2.使用ViewHolder

ViewHolder将需要缓存的view封装好,convertView的setTag才是将这些缓存起来供下次调用。 当你的listview里布局多样化的时候 viewholder的作用体现明显,效率再一次提高。 View的findViewById()方法也是比较耗时的,因此需要考虑只调用一次,之后就用View.getTag()方法来获得ViewHolder对象。

3.优雅的使用ViewHolder

使用ViewHolder时,每次一遍一遍的findViewById,一遍一遍在ViewHolder里面添加View的定义,view一多,是不是感觉烦爆了,base-adapter-helper这个类库似乎完美的解决了这个问题。

其设计思想是使用 SparseArray来存储view的引用,代替了原本的ViewHolder,不用声明一大堆View,简洁明了。

(1).ViewHolder类的封装如下:ViewHolder类:

package com.chengdong.su.baseadapter.util;

import android.content.Context;
import android.util.Log;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;

/**
 * the common object of the ViewHolder
 * 
 * @author scd
 * 
 */
public class ViewHolder {
    /** the object of the TAG */
    private String TAG = getClass().getSimpleName();
    /** the object of the view */
    private SparseArray mViews;
    /** the object of the position */
    private int mPosition;
    /** the object of the converview */
    private View mConvertView;

    /***
     * 构造方法
     * 
     * @param context
     * @param parent
     * @param layoutId
     * @param position
     */
    public ViewHolder(Context context, ViewGroup parent, int layoutId,int position) {
      super();
      this.mPosition = position;
      this.mViews = new SparseArray();
      mConvertView = LayoutInflater.from(context).inflate(layoutId, parent,
                false);
        mConvertView.setTag(this);

    }

    /**
     * get the object of the ViewHolder
     * 
     * @param context
     * @param convertView
     * @param parent
     * @param layoutId
     * @param position
     * @return
     */
    public static ViewHolder getViewHolder(Context context, View convertView,
            ViewGroup parent, int layoutId, int position) {
        if (convertView == null) {
            return new ViewHolder(context, parent, layoutId, position);
        } else {
            ViewHolder holder = (ViewHolder) convertView.getTag();
            // 修改位置变化
            holder.mPosition = position;
            Log.d("getViewHolder", "getViewHolder:position:--->" + position);
            return holder;
        }
    }

    /**
     * find the view by the viewId
     * 
     * @param viewId
     * @return
     */
    @SuppressWarnings("unchecked")
    public  T findView(int viewId) {
        View view = mViews.get(viewId);
        if (view == null) {
            view = mConvertView.findViewById(viewId);
            // 添加View对象
            mViews.put(viewId, view);

        }
        return (T) view;
    }

    /**
     * get the object of the convertView
     * 
     * @return
     */
    public View getConvertView() {
        return mConvertView;
    }

    public int getPosition() {
        return mPosition;
    }

    /**
     * 为TextView设置值
     * 
     * @param viewId
     * @param text
     * @return
     */
    public ViewHolder setText(int viewId, String text) {
        TextView textView = findView(viewId);
        textView.setText(text);
        return this;
    }

    /**
     * 为ImageView设置值
     * 
     * @param viewId
     * @param resId
     * @return
     */
    public ViewHolder setImageResource(int viewId, int resId) {
        ImageView view = findView(viewId);
        view.setImageResource(resId);
        return this;
    }

    // TODO 待写View设置的辅助的方法

}

(2).CommonAdapter共通类的实现如下:

package com.chengdong.su.baseadapter.util;

import java.util.List;

import android.R.integer;
import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;

import com.chengdong.su.baseadapter.R;

public abstract class CommonAdapter<T> extends BaseAdapter {
    /** 上下文 */
    protected Context mContext;
    /** 数据源 */
    protected List mData;
    private int mLayoutId;

    public CommonAdapter(Context mContext, int layoutId) {
        super();
        this.mContext = mContext;
        this.mLayoutId = layoutId;
    }

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

    @Override
    public T getItem(int position) {
        return (T) (mData == null ? 0 : mData.get(position));
    }

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

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {

        ViewHolder holder = ViewHolder.getViewHolder(mContext, convertView,
                parent, mLayoutId, position);
        convert(holder, getItem(position));
        return holder.getConvertView();

    }

    /*** 设置数据 */
    public abstract void setData(List data);

    /*** 实现给View赋数据的方法 */
    public abstract void convert(ViewHolder holder, T item);
}

(3).MyAdapter类的实现:

package com.chengdong.su.baseadapter.adapter;

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

import android.content.Context;
import android.view.View;
import android.widget.CheckBox;

import com.chengdong.su.baseadapter.R;
import com.chengdong.su.baseadapter.bean.Bean;
import com.chengdong.su.baseadapter.util.CommonAdapter;
import com.chengdong.su.baseadapter.util.ViewHolder;

/**
 * 通用的ViewHolder
 * 
 * @author scd
 * 
 */
public class MyAdapter extends CommonAdapter {
    private List mList = new ArrayList();

    public MyAdapter(Context mContext) {
        super(mContext, R.layout.item_listview);
    }

    @Override
    public void setData(List data) {
        mData = data;
        notifyDataSetChanged();

    }

    @Override
    public void convert(final ViewHolder holder, Bean item) {
        // 链式编程
        holder.setText(R.id.tv_title, item.getTitle())
                .setText(R.id.tv_desc, item.getDesc())
                .setText(R.id.tv_time, item.getTime())
                .setText(R.id.tv_phone, item.getPhone());

        // checkBox复用导致内容错乱的解决方法
        final CheckBox checkBox = holder.findView(R.id.cb_select);
        // // 默认状态下都不选中
        checkBox.setChecked(false);
        if (mList.contains(holder.getPosition())) {
            checkBox.setChecked(true);
        }
        checkBox.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                if (checkBox.isChecked()) {
                    mList.add((Integer) holder.getPosition());

                } else {
                    mList.remove((Integer) holder.getPosition());
                }

            }
        });

    }

}

(4)MainActivity类的实现如下:

package com.chengdong.su.baseadapter;

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

import com.chengdong.su.baseadapter.adapter.MyAdapter;
import com.chengdong.su.baseadapter.bean.Bean;

import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;

public class MainActivity extends Activity {
    private ListView mListView;
    private MyAdapter mAdapter;
    private List mData;

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

    }

    private void initView() {
        mListView = (ListView) findViewById(R.id.lv_show);
        mAdapter = new MyAdapter(MainActivity.this);
        initData();
        mAdapter.setData(mData);
        mListView.setAdapter(mAdapter);

    }

    private void initData() {
        mData = new ArrayList();
        Bean bean = new Bean("1", "2", "3", "4");
        for (int i = 0; i < 11; i++) {

            mData.add(bean);
        }

    }

}

(5).activity_main类的实现如下:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

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

LinearLayout>

(6).item_listview类的实现:


<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:descendantFocusability="blocksDescendants"
    android:orientation="vertical"
    android:padding="8dp" >

    <CheckBox
        android:id="@+id/cb_select"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentRight="true"
        android:layout_centerVertical="true"
        android:focusable="false" />

    <TextView
        android:id="@+id/tv_title"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_toLeftOf="@id/cb_select"
        android:maxLines="1"
        android:text="标题" />

    <TextView
        android:id="@+id/tv_desc"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/tv_title"
        android:layout_marginTop="8dp"
        android:maxLines="1"
        android:text="描述" />


    <TextView
        android:id="@+id/tv_time"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/tv_desc"
        android:layout_marginTop="8dp"
        android:text="时间" />

    <TextView
        android:id="@+id/tv_phone"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentRight="true"
        android:layout_below="@id/tv_desc"
        android:layout_marginTop="8dp"
        android:text="电话" />


RelativeLayout>

说明:
android:descendantFocusability="blocksDescendants"
android:focusable="false"

是当存在checkbox或者其他控件存在导致item抢占焦点的解决方案:解决item不能被选中点击。
// checkBox复用导致内容错乱的解决方法如下:

 final CheckBox checkBox = holder.findView(R.id.cb_select);
        // // 默认状态下都不选中
        checkBox.setChecked(false);
        if (mList.contains(holder.getPosition())) {
            checkBox.setChecked(true);
        }
        checkBox.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View v) {
                if (checkBox.isChecked()) {
                    mList.add((Integer) holder.getPosition());

                } else {
                    mList.remove((Integer) holder.getPosition());
                }

            }
        });

你可能感兴趣的:(app源码)