[Android]对BaseAdapter中ViewHolder编写简化

简介:

以下内容为原创,欢迎转载,转载请注明

来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/3642849.html

 

在Android项目中,经常都会用到ListView这个控件,而相应的Adapter中getView()方法的编写有一个标准的形式,如下:

复制代码
 1 @Override
 2     public View getView(int position, View convertView, ViewGroup parent) {
 3         ViewHolder holder;
 4         if(null == convertView){
 5             holder = new ViewHolder();
 6             LayoutInflater mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
 7             convertView = mInflater.inflate(R.layout.item, null);
 8             holder.btn = (Button) convertView.findViewById(R.id.btn);
 9             holder.tv = (TextView) convertView.findViewById(R.id.tv);
10             holder.iv = (TextView) convertView.findViewById(R.id.iv);
11 
12             convertView.setTag(holder);
13         }else{
14             holder = (ViewHolder) convertView.getTag();
15         }
16         final HashMap<String, Object> map = list.get(position);
17 
18         holder.iv.setImageResource(Integer.valueOf(map.get("iv").toString()));
19         holder.tv.setText(map.get("tv").toString());
20 
21         holder.btn.setOnClickListener(new View.OnClickListener() {
22             @Override
23             public void onClick(View v) {
24                 Toast.makeText(context, map.get("btn").toString(), Toast.LENGTH_SHORT).show();
25             }
26         });
27 
28         return convertView;
29     }
30 
31     class ViewHolder{
32         Button btn;
33         ImageView iv;
34         TextView tv;
35 
36     }
复制代码

以下是碎碎念(想直接看代码的,就跳过这段吧-_-!):

也就是说每次编写Adapter都需要编写class ViewHolder...、if(null == convertView){...等等。这些代码跟业务逻辑关系不大,没有必要每次都写重复代码,

所以,显然有简化代码的余地。

既然我们的需求是不需要重复编写ViewHolder等内部类,那就把它移到父类吧。

但是ViewHolder中在实际项目中有不同的View,那就用list存放起来吧,但是放在list中的话,怎么取出来?用index?显然不是好的方法,不是有Resource Id这玩意,通过这个取不就好了么?所以以Resource Id为key放在Map中比较合适,但是既然以int(Resource Id)为key,那自然而然想到使用SparseArray了。

然后再把if(null == converView)...这些代码统统移到父类中。

所以ABaseAdapter诞生了,代码如下:

复制代码
 1 /**
 2  * 实现对BaseAdapter中ViewHolder相关的简化
 3  * Created with IntelliJ IDEA.
 4  * Author: wangjie  email:tiantian.china.2@gmail.com
 5  * Date: 14-4-2
 6  * Time: 下午5:54
 7  */
 8 public abstract class ABaseAdapter extends BaseAdapter{
 9     Context context;
10 
11     protected ABaseAdapter(Context context) {
12         this.context = context;
13     }
14 
15     protected ABaseAdapter() {
16     }
17 
18     /**
19      * 各个控件的缓存
20      */
21     public class ViewHolder{
22         public SparseArray<View> views = new SparseArray<View>();
23 
24         /**
25          * 指定resId和类型即可获取到相应的view
26          * @param convertView
27          * @param resId
28          * @param <T>
29          * @return
30          */
31         public <T extends View> T obtainView(View convertView, int resId){
32             View v = views.get(resId);
33             if(null == v){
34                 v = convertView.findViewById(resId);
35                 views.put(resId, v);
36             }
37             return (T)v;
38         }
39 
40     }
41 
42     /**
43      * 改方法需要子类实现,需要返回item布局的resource id
44      * @return
45      */
46     public abstract int itemLayoutRes();
47 
48     @Override
49     public View getView(int position, View convertView, ViewGroup parent) {
50         ViewHolder holder;
51         if(null == convertView){
52             holder = new ViewHolder();
53             convertView = LayoutInflater.from(context).inflate(itemLayoutRes(), null);
54             convertView.setTag(holder);
55         }else{
56             holder = (ViewHolder) convertView.getTag();
57         }
58         return getView(position, convertView, parent, holder);
59     }
60 
61     /**
62      * 使用该getView方法替换原来的getView方法,需要子类实现
63      * @param position
64      * @param convertView
65      * @param parent
66      * @param holder
67      * @return
68      */
69     public abstract View getView(int position, View convertView, ViewGroup parent, ViewHolder holder);
70 
71 }
复制代码

如上代码:增加了一个itemLayoutRes()的抽象方法,该抽象方法提供给子类实现,返回item布局的resource id,为后面的getView方法提供调用。

可以看到上述代码中,在系统的getView方法中,进行我们以前在BaseAdapter实现类中所做的事(初始化converView,并绑定ViewHolder作为tag),然后抛弃了系统提供的getView方法,直接去调用自己写的getView抽象方法,这个getView方法是提供给子类去实现的,作用跟系统的getView一样,但是这个getView方法中携带了一个ViewHolder对象,子类可以通过这个对象进行获取item中的控件。

具体使用方式如下,比如创建了MyAdapter并继承了ABaseAdapter:

注意,在构造方法中需要调用父类的构造方法:

1 public MyAdapter(Context context, List<HashMap<String, Object>> list) {
2     super(context);
3     this.list = list;
4 }

即,以上的“super(context);”必须调用。

接着实现itemLayoutRes()方法,返回item的布局:

1 @Override
2 public int itemLayoutRes() {
3     return R.layout.item;
4 }

 

getView方法中的实现如下:

复制代码
 1 @Override
 2 public View getView(int position, View convertView, ViewGroup parent, ViewHolder holder) {
 3     final HashMap<String, Object> map = list.get(position);
 4 
 5     Button btn = holder.obtainView(convertView, R.id.item_btn);
 6     ImageView iv = holder.obtainView(convertView, R.id.item_iv);
 7     TextView tv = holder.obtainView(convertView, R.id.item_tv);
 8 
 9      btn.setOnClickListener(new View.OnClickListener() {
10         @Override
11         public void onClick(View v) {
12             Toast.makeText(context, map.get("btn").toString(), Toast.LENGTH_SHORT).show();
13          }
14      });
15 
16      iv.setImageResource(Integer.valueOf(map.get("iv").toString()));
17      tv.setText(map.get("tv").toString());
18 
19         return convertView;
20     }
复制代码

如上代码:调用holder.obtainView方法既可获取item中的控件;

 

 

本文转自天天_byconan博客园博客,原文链接:http://www.cnblogs.com/tiantianbyconan/p/3642849.html ,如需转载请自行联系原作者
相关文章
|
3月前
|
存储 Android开发 开发者
Android项目架构设计问题之定义RecyclerView的ViewHolder如何解决
Android项目架构设计问题之定义RecyclerView的ViewHolder如何解决
41 0
|
6月前
|
XML Java Android开发
Android Studio App开发之列表类视图中基本适配器BaseAdapter的使用及实战(附源码 超详细)
Android Studio App开发之列表类视图中基本适配器BaseAdapter的使用及实战(附源码 超详细)
203 0
|
Ubuntu 物联网 Android开发
QT应用编程: 编写低功耗BLE蓝牙调试助手(Android系统APP)
QT应用编程: 编写低功耗BLE蓝牙调试助手(Android系统APP)
1565 0
QT应用编程: 编写低功耗BLE蓝牙调试助手(Android系统APP)
|
Android开发
android用异步操作AsyncTask编写文件查看器
android用异步操作AsyncTask编写文件查看器
129 0
|
Android开发 Python
【Android 逆向】使用 Python 编写 APK 批处理分析工具
【Android 逆向】使用 Python 编写 APK 批处理分析工具
460 0
【Android 逆向】使用 Python 编写 APK 批处理分析工具
|
安全 Java Android开发
【Android 安装包优化】开启 ProGuard 混淆 ( 压缩 Shrink | 优化 Optimize | 混淆 Obfuscate | 预检 | 混淆文件编写 | 混淆前后对比 )
【Android 安装包优化】开启 ProGuard 混淆 ( 压缩 Shrink | 优化 Optimize | 混淆 Obfuscate | 预检 | 混淆文件编写 | 混淆前后对比 )
287 0
【Android 安装包优化】开启 ProGuard 混淆 ( 压缩 Shrink | 优化 Optimize | 混淆 Obfuscate | 预检 | 混淆文件编写 | 混淆前后对比 )
|
Android开发 C++
【Android 内存优化】libjpeg-turbo 函数库交叉编译与使用 ( 交叉编译脚本编写 | 函数库头文件拷贝 | 构建脚本配置 | Android Studio 测试函数库 )(二)
【Android 内存优化】libjpeg-turbo 函数库交叉编译与使用 ( 交叉编译脚本编写 | 函数库头文件拷贝 | 构建脚本配置 | Android Studio 测试函数库 )(二)
332 0
【Android 内存优化】libjpeg-turbo 函数库交叉编译与使用 ( 交叉编译脚本编写 | 函数库头文件拷贝 | 构建脚本配置 | Android Studio 测试函数库 )(二)
|
Ubuntu Shell Linux
【Android 内存优化】libjpeg-turbo 函数库交叉编译与使用 ( 交叉编译脚本编写 | 函数库头文件拷贝 | 构建脚本配置 | Android Studio 测试函数库 )(一)
【Android 内存优化】libjpeg-turbo 函数库交叉编译与使用 ( 交叉编译脚本编写 | 函数库头文件拷贝 | 构建脚本配置 | Android Studio 测试函数库 )(一)
425 0
【Android 内存优化】libjpeg-turbo 函数库交叉编译与使用 ( 交叉编译脚本编写 | 函数库头文件拷贝 | 构建脚本配置 | Android Studio 测试函数库 )(一)
|
网络协议 Ubuntu Android开发
QT应用编程: 编写网络调试助手(Android系统APP)
QT应用编程: 编写网络调试助手(Android系统APP)
336 0
QT应用编程: 编写网络调试助手(Android系统APP)
|
Ubuntu Android开发
QT应用编程: 编写HC05串口蓝牙调试助手(Android系统APP)
QT应用编程: 编写HC05串口蓝牙调试助手(Android系统APP)
664 0
QT应用编程: 编写HC05串口蓝牙调试助手(Android系统APP)