其实,在我开发第一个app的时候,我已经对adapter的重写已经感觉到厌烦了,为什么每一个ListView都要写一个adapter,而且都是那么的耗时间,而且每一个ListView或者GridView都要创建一个新的adapter类,当时一个app用至少20个item不重复的ListView或者GridView,理所当然的,重写了20多个adapter类,这些都是用时间写出来的。但是,当时技术有限,一直有一个问题困扰着我,使我无法真正完成一个万能adapter;
每一个adapter,都会写一个ViewHolder类,这是为了减少使用findViewById次数的解决办法,打造一个万能的adpter,这个问题就成了我的拦路虎了。怎么样才能只有在需要的时候才创建ViewHolder的成员变量呢?
直到我看到一鸿洋的一篇文章,我终于找到了一个解决办法,那篇文章的地址是:http://blog.csdn.net/lmj623565791/article/details/38902805/
有了鸿洋的通用ViewHodler之后,万能适配的制作就变得简单了
一般情况,我们重写adapter的时候,getView肯定会出现的代码段如下
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder=null;
if(convertView==null){
convertView=mInflater.inflate(mItemLayoutId,null);
holder=new ViewHolder();
//这里进行一系列的findViewById方法
convertView.setTag(holder);
}else{
holder= (ViewHolder) convertView.getTag();
}
return convertView;
}
首先,我根据鸿洋的通用ViewHolder进行一点点的修改,修改结果如下:
protected class ViewHolder {
private final SparseArray<View> mViews;//性能优化,使用SparseArray代替HashMap
private View mConvertView;
public ViewHolder(View view) {
this.mViews = new SparseArray<View>();
this.mConvertView = view;
}
/**
* 通过控件的Id获取对于的控件,如果没有则加入views
*
* @param viewId
* @return
*/
public <T extends View> T getView(int viewId) {
View view = mViews.get(viewId);
if (view == null) {
view = mConvertView.findViewById(viewId);
mViews.put(viewId, view);
}
return (T) view;
}
/**
* 为TextView设置字符串
*
* @param viewId
* @param text
* @return
*/
public ViewHolder setText(int viewId, String text)
{
TextView view = getView(viewId);
view.setText(text);
return this;
}
}
粗略一看可能没啥区别,但是也确实没多大区别,唯一区别是,我进行了简化,说白了,觉得没用的方法进行了删除。
然后,就是改造adapter的getView方法里面的内容了,结果如下:
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder=null;
if(convertView==null){
convertView=mInflater.inflate(mItemLayoutId,null);
holder=new ViewHolder(convertView);
convertView.setTag(holder);
}else{
holder= (ViewHolder) convertView.getTag();
}
convert(holder, getItem(position),position);
return convertView;
}
public abstract void convert(ViewHolder helper, T item,int position);
这里,使用convert()抽象方法,让getview里要处理的事件,在外部回调进来。
OK,这个完成之后,剩下的就不在是甚么问题了,无非就是数据的传输而已,
万能Adapter的完整类我贴给大家
package com.loyalty.Common.base;
import android.content.Context;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
import java.util.ArrayList;
import java.util.List;
/**
* 万能适配器
* Created by Jin on 2015/9/10.
* desc:
*/
public abstract class ListAdapter<T> extends BaseAdapter {
private LayoutInflater mInflater;
private List<T> mLists = new ArrayList<T>();
private int mItemLayoutId = 0;
public ListAdapter(Context context, int itemLayoutId) {
mInflater = LayoutInflater.from(context);
this.mItemLayoutId = itemLayoutId;
}
/**
* 获取当前列表数据
* @return
*/
public List<T> getCurData() {
return mLists;
}
/**
* 设置最新数据
* @param lists
*/
public void setData(List<T> lists) {
mLists.clear();
this.mLists.addAll(lists);
notifyDataSetChanged();
}
/**
* 在当前列表的基础上添加新的数据
* @param lists
*/
public void addData(List<T> lists){
this.mLists.addAll(lists);
notifyDataSetChanged();
}
/**
* 在某个位置插入一条数据
* @param ss
* @param position
*/
public void addItem(T ss,int position){
mLists.add(position,ss);
notifyDataSetChanged();
}
/**
* 删除一条数据
* @param position
*/
public void delItem(int position){
mLists.remove(position);
notifyDataSetChanged();
}
@Override
public int getCount() {
return mLists.size();
}
@Override
public T getItem(int position) {
return mLists.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder=null;
if(convertView==null){
convertView=mInflater.inflate(mItemLayoutId,null);
holder=new ViewHolder(convertView);
convertView.setTag(holder);
}else{
holder= (ViewHolder) convertView.getTag();
}
convert(holder, getItem(position),position);
return convertView;
}
public abstract void convert(ViewHolder helper, T item,int position);
protected class ViewHolder {
private final SparseArray<View> mViews;//性能优化,使用SparseArray代替HashMap
private View mConvertView;
public ViewHolder(View view) {
this.mViews = new SparseArray<View>();
this.mConvertView = view;
}
/**
* 通过控件的Id获取对于的控件,如果没有则加入views
*
* @param viewId
* @return
*/
public <T extends View> T getView(int viewId) {
View view = mViews.get(viewId);
if (view == null) {
view = mConvertView.findViewById(viewId);
mViews.put(viewId, view);
}
return (T) view;
}
/**
* 为TextView设置字符串
*
* @param viewId
* @param text
* @return
*/
public ViewHolder setText(int viewId, String text)
{
TextView view = getView(viewId);
view.setText(text);
return this;
}
}
}
最后说点话,我为什么不把数据通过adapter类的构造器传进来,在我写的项目经验中,很多时候在ListView或者GridView调用setAdapter()方法的时候,数据还没有有加载好,所以很没有必要给适配器多加这么一个参数。当然各有各的写法,怎么习惯怎么来就是了。o(∩_∩)o 哈哈
至于用法如下:
package com.jin.jadapter;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.ListView;
import android.widget.Toast;
import com.bumptech.glide.Glide;
import com.jin.common.base.CommonAdapter;
import java.util.ArrayList;
import java.util.List;
import de.hdodenhof.circleimageview.CircleImageView;
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
private ListView mLvMain;
private CommonAdapter mMainAdapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
init();
}
private List<Item> data() {
List<Item> lists=new ArrayList<Item>();
Item item=new Item();
item.setTitle("杨幂").setUrl("http://img2.imgtn.bdimg.com/it/u=3289766452,1285249418&fm=21&gp=0.jpg");
Item item1=new Item();
item1.setTitle("郑爽").setUrl("http://img1.imgtn.bdimg.com/it/u=1375490053,1644511488&fm=21&gp=0.jpg");
Item item2=new Item();
item2.setTitle("戚薇").setUrl("http://img2.imgtn.bdimg.com/it/u=806822895,2143642379&fm=21&gp=0.jpg");
Item item3=new Item();
item3.setTitle("刘亦菲").setUrl("http://img2.imgtn.bdimg.com/it/u=2625445638,3608888874&fm=21&gp=0.jpg");
lists.add(item);
lists.add(item1);
lists.add(item2);
lists.add(item3);
return lists;
}
private void init() {
mLvMain=(ListView)findViewById(R.id.lv_main);
mMainAdapter=new CommonAdapter(this,R.layout.item_lv_main) {
@Override
public void convert(ViewHolder helper, Object item, int position) {
Item item1= (Item) item;
helper.setText(R.id.tx_item_main_name,item1.getTitle());//设置textview的内容
Glide.with(MainActivity.this).load(item1.getUrl()).into(((CircleImageView) helper.getView(R.id.iv_item_main_head)));//设置Imageview的内容
helper.getView(R.id.btn_item_main_more).setOnClickListener(new View.OnClickListener() {//设置button的监听时间
@Override
public void onClick(View v) {
Toast.makeText(MainActivity.this,"没有更多",Toast.LENGTH_SHORT).show();
}
});
}
};
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.btn_main_refresh:
mMainAdapter.setData(data());//刷新数据
break;
case R.id.btn_main_more:
mMainAdapter.addData(data());//加载更多
break;
}
}
}
真相:
源码地址:http://download.csdn.net/detail/kimyo8337373/9151827