Android 快速开发系列 打造万能的ListView GridView 适配器

this.mViews = new SparseArray();

mConvertView = LayoutInflater.from(context).inflate(layoutId, parent,

false);

//setTag

mConvertView.setTag(this);

}

/**

  • 拿到一个ViewHolder对象

  • @param context

  • @param convertView

  • @param parent

  • @param layoutId

  • @param position

  • @return

*/

public static ViewHolder get(Context context, View convertView,

ViewGroup parent, int layoutId, int position)

{

if (convertView == null)

{

return new ViewHolder(context, parent, layoutId, position);

}

return (ViewHolder) convertView.getTag();

}

/**

  • 通过控件的Id获取对于的控件,如果没有则加入views

  • @param viewId

  • @return

*/

public T getView(int viewId)

{

View view = mViews.get(viewId);

if (view == null)

{

view = mConvertView.findViewById(viewId);

mViews.put(viewId, view);

}

return (T) view;

}

public View getConvertView()

{

return mConvertView;

}

}

与传统的ViewHolder不同,我们使用了一个SparseArray用于存储与之对于的convertView的所有的控件,当需要拿这些控件时,通过getView(id)进行获取;

下面看使用该ViewHolder的MyAdapter;

@Override

public View getView(int position, View convertView, ViewGroup parent)

{

//实例化一个viewHolder

ViewHolder viewHolder = ViewHolder.get(mContext, convertView, parent,

R.layout.item_single_str, position);

//通过getView获取控件

TextView tv = viewHolder.getView(R.id.id_tv_title);

//使用

tv.setText(mDatas.get(position));

return viewHolder.getConvertView();

}

只看getView,其他方法都一样;首先调用ViewHolder的get方法,如果convertView为null,new一个ViewHolder实例,通过使用mInflater.inflate加载布局,然后new一个SparseArray用于存储View,最后setTag(this);

如果存在那么直接getTag

最后通过getView(id)获取控件,如果存在则直接返回,否则调用findViewById,返回存储,返回。

好了,一个通用的ViewHolder写好了,以后一个项目几十个Adapter一个ViewHolder直接hold住全场大家可以省点时间斗个小地主了

4、打造通用的Adapter

==============

有了通用的ViewHolder大家肯定不能满足,怎么也得省出dota的时间,人在塔在~~

下面看如何打造一个通过的Adapter,我们叫做CommonAdapter

继续分析,Adapter一般需要保持一个List对象,存储一个Bean的集合,不同的ListView,Bean肯定是不同的,这个CommonAdapter肯定需要支持泛型,内部维持一个List,就解决我们的问题了;

于是我们初步打造我们的CommonAdapter

package com.example.zhy_baseadapterhelper;

import java.util.List;

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 abstract class CommonAdapter extends BaseAdapter

{

protected LayoutInflater mInflater;

protected Context mContext;

protected List mDatas;

public CommonAdapter(Context context, List mDatas)

{

mInflater = LayoutInflater.from(context);

this.mContext = context;

this.mDatas = mDatas;

}

@Override

public int getCount()

{

return mDatas.size();

}

@Override

public Object getItem(int position)

{

return mDatas.get(position);

}

@Override

public long getItemId(int position)

{

return position;

}

}

我们的CommonAdapter依然是一个抽象类,除了getView以外我们把其他的代码都实现了,这样的话,在使用我们的Adapter只要实现一个getView,然后getView里面再使用我们打造的通过的ViewHolder是不是感觉还不错~

现在我们的MyAdapter是这样的:

package com.example.zhy_baseadapterhelper;

import java.util.List;

import android.content.Context;

import android.view.View;

import android.view.ViewGroup;

import android.widget.TextView;

public class MyAdapter extends CommonAdapter

{

public MyAdapter(Context context, List mDatas)

{

super(context, mDatas);

}

@Override

public View getView(int position, View convertView, ViewGroup parent)

{

ViewHolder viewHolder = ViewHolder.get(mContext, convertView, parent,

R.layout.item_single_str, position);

TextView mTitle = viewHolder.getView(R.id.id_tv_title);

mTitle.setText((String) mDatas.get(position));

return viewHolder.getConvertView();

}

}

所有的代码加起来也就10行左右,是不是神清气爽~~稍等,我先去dota一把~

但是我们是否就这样满足了呢?显然还可以简化。

5、进一步铸造

=======

注意我们的getView里面的代码,虽然只有4行,但是我觉得所有的Adapter的

第一行(ViewHolder viewHolder = getViewHolder(position, convertView,parent);)和

最后一行:return viewHolder.getConvertView();一定是一样的。

那么我们可以这样做:我们把第一行和最后一行写死,把中间变化的部分抽取出来,这不就是OO的设计原则嘛。现在CommonAdapter是这样的:

package com.example.zhy_baseadapterhelper;

import java.util.List;

import android.content.Context;

import android.view.LayoutInflater;

import android.view.View;

import android.view.ViewGroup;

import android.widget.BaseAdapter;

public abstract class CommonAdapter extends BaseAdapter

{

protected LayoutInflater mInflater;

protected Context mContext;

protected List mDatas;

protected final int mItemLayoutId;

public CommonAdapter(Context context, List mDatas, int itemLayoutId)

{

this.mContext = context;

this.mInflater = LayoutInflater.from(mContext);

this.mDatas = mDatas;

this.mItemLayoutId = itemLayoutId;

}

@Override

public int getCount()

{

return mDatas.size();

}

@Override

public T 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)

{

final ViewHolder viewHolder = getViewHolder(position, convertView,

parent);

convert(viewHolder, getItem(position));

return viewHolder.getConvertView();

}

public abstract void convert(ViewHolder helper, T item);

private ViewHolder getViewHolder(int position, View convertView,

ViewGroup parent)

{

return ViewHolder.get(mContext, convertView, parent, mItemLayoutId,

position);

}

}

对外公布了一个convert方法,并且还把viewHolder和本Item对于的Bean对象给传出去,现在convert方法里面需要干嘛呢?

通过ViewHolder把View找到,通过Item设置值;

现在我觉得代码简化到这样,我已经不需要单独写一个Adapter了,直接MainActivity匿名内部类走起~

@Override

protected void onCreate(Bundle savedInstanceState)

{

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

mListView = (ListView) findViewById(R.id.id_lv_main);

//设置适配器

mListView.setAdapter(mAdapter = new CommonAdapter(

getApplicationContext(), mDatas, R.layout.item_single_str)

{

@Override

public void convert(ViewHolder c, String item)

{

TextView view = viewHolder.getView(R.id.id_tv_title);

view.setText(item);

}

});

}

可以看到效果咋样,不错吧。你觉得还能简化么?我觉得还能改善。

6、Adapter最后的封魔

==============

我们现在在convertView里面需要这样:

@Override

public void convert(ViewHolder viewHolder, String item)

{

TextView view = viewHolder.getView(R.id.id_tv_title);

view.setText(item);

}

我们细想一下,其实布局里面的View常用也就那么几种:ImageView,TextView,Button,CheckBox等等;

那么我觉得ViewHolder还可以封装一些常用的方法,比如setText(id,String);setImageResource(viewId, resId);setImageBitmap(viewId, bitmap);

那么现在ViewHolder是:

package com.example.zhy_baseadapterhelper;

import android.content.Context;

import android.graphics.Bitmap;

import android.util.SparseArray;

import android.view.LayoutInflater;

import android.view.View;

import android.view.ViewGroup;

import android.widget.ImageView;

import android.widget.TextView;

import com.example.zhy_baseadapterhelper.ImageLoader.Type;

public class ViewHolder

{

private final SparseArray mViews;

private int mPosition;

private View mConvertView;

private ViewHolder(Context context, ViewGroup parent, int layoutId,

int position)

{

this.mPosition = position;

this.mViews = new SparseArray();

mConvertView = LayoutInflater.from(context).inflate(layoutId, parent,

false);

// setTag

mConvertView.setTag(this);

}

/**

  • 拿到一个ViewHolder对象

  • @param context

  • @param convertView

  • @param parent

  • @param layoutId

  • @param position

  • @return

*/

public static ViewHolder get(Context context, View convertView,

ViewGroup parent, int layoutId, int position)

{

if (convertView == null)

{

return new ViewHolder(context, parent, layoutId, position);

}

return (ViewHolder) convertView.getTag();

}

public View getConvertView()

{

return mConvertView;

}

/**

  • 通过控件的Id获取对于的控件,如果没有则加入views

  • @param viewId

  • @return

*/

public 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;

}

/**

  • 为ImageView设置图片

  • @param viewId

  • @param drawableId

  • @return

*/

public ViewHolder setImageResource(int viewId, int drawableId)

{

ImageView view = getView(viewId);

view.setImageResource(drawableId);

return this;

}

/**

  • 为ImageView设置图片

  • @param viewId

  • @param drawableId

  • @return

*/

public ViewHolder setImageBitmap(int viewId, Bitmap bm)

{

ImageView view = getView(viewId);

view.setImageBitmap(bm);

return this;

}

/**

  • 为ImageView设置图片

  • @param viewId

  • @param drawableId

  • @return

*/

public ViewHolder setImageByUrl(int viewId, String url)

{

ImageLoader.getInstance(3, Type.LIFO).loadImage(url,

(ImageView) getView(viewId));

return this;

}

public int getPosition()

{

return mPosition;

}

}

现在的MainActivity只需要这么写:

mAdapter = new CommonAdapter(getApplicationContext(),

R.layout.item_single_str, mDatas)

{

@Override

protected void convert(ViewHolder viewHolder, String item)

{

viewHolder.setText(R.id.id_tv_title, item);

}

};

convertView里面只要一行代码了~~~

好了,到此我们的通用的Adapter已经一步一步铸造完毕咋样,以后写项目省下来的时间是不是可以陪我切磋dota了(ps:11昵称:血魔哥404)~

注:关于ViewHolder里面的setText,setImageResource这类的方法,大家可以在使用的过程中不断的完善,今天发现这个控件可以这么设置值,好,放进去;时间长了,基本就完善了。还有那个ImageLoader是我另一篇博客里的,大家可以使用UIL,Volley或者自己写个图片加载器;

7、实践

====

说了这么多,还是得拿出来让我们的实践检验检验,顺便来几张套图,俗话说,没图没正相。

1、我们的实例代码的图是这样的:

关于Adapter和ViewHolder的代码是这样的:

// 设置适配器

mListView.setAdapter(mAdapter = new CommonAdapter(

getApplicationContext(), mDatas, R.layout.item_single_str)

{

@Override

public void convert(ViewHolder helper, String item)

{

helper.setText(R.id.id_tv_title,item);

}

});

哎哟,我是不是只要贴一行;

2、来个复杂点的布局

<?xml version="1.0" encoding="utf-8"?>

<RelativeLayout xmlns:android=“http://schemas.android.com/apk/res/android”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:background=“#ffffff”

android:orientation=“vertical”

android:padding=“10dp” >

<TextView

android:id=“@+id/tv_title”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:singleLine=“true”

android:text=“红色钱包”

android:textSize=“16sp”

android:textColor=“#444444” >

<TextView

android:id=“@+id/tv_describe”

android:layout_width=“match_parent”

android:layout_height=“wrap_content”

android:layout_below=“@id/tv_title”

android:layout_marginTop=“10dp”

android:maxLines=“2”

android:minLines=“1”

android:text=“周三早上丢失了红色钱包,在食堂二楼”

android:textColor=“#898989”

android:textSize=“16sp” >

<TextView

android:id=“@+id/tv_time”

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

在这里小编整理了一份Android大厂常见面试题,和一些Android架构视频解析,都已整理成文档,全部都已打包好了,希望能够对大家有所帮助,在面试中能顺利通过。

image

image

喜欢本文的话,不妨顺手给我点个小赞、评论区留言或者转发支持一下呗

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!

_content"

android:layout_height=“wrap_content”
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

[外链图片转存中…(img-gT9kiUec-1712159327878)]

[外链图片转存中…(img-tceep8VJ-1712159327879)]

[外链图片转存中…(img-QlWPS6go-1712159327879)]

[外链图片转存中…(img-45yoTjqk-1712159327880)]

[外链图片转存中…(img-PAfkDzfR-1712159327880)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

在这里小编整理了一份Android大厂常见面试题,和一些Android架构视频解析,都已整理成文档,全部都已打包好了,希望能够对大家有所帮助,在面试中能顺利通过。

[外链图片转存中…(img-toJXmJa1-1712159327880)]

[外链图片转存中…(img-r8USs5vS-1712159327881)]

喜欢本文的话,不妨顺手给我点个小赞、评论区留言或者转发支持一下呗

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
  • 26
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值