在写ViewList的时候要写Adapter的时候,经常大量的代码都是差不多的。

1 ViewHold

2 if(convertView ==null ){}else{}

3 setTag

4 FIndElement 和 Set 等等

所以我想能不能写一个通用的ViewHold的和通用的Adapter呢?

我们来试试吧。

直接上代码,然后再进行分析。

<?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/title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="20pt"
/>
<TextView
android:id="@+id/context"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="15pt"/> </LinearLayout>

Bean数据结构

public class Bean {
String context;
String title; public String getTitle() {
return title;
} public void setTitle(String title) {
this.title = title;
} public String getContext() {
return context;
} public void setContext(String context) {
this.context = context;
}
}

通用的ViewHold

import android.content.Context;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup; /**
* Created by sunfan on 15-4-19.
* 通用的ViewHold类
*/
public class CommonViewHold {
private View mView;//当前页面
private SparseArray<View> elements = new SparseArray<View>();//控件元素 /**
* 构造函数,当传入的convertView为null时对mView进行初始化,并设置tag
* @param context
* @param parent
* @param resource
*/
private CommonViewHold(Context context, ViewGroup parent, int resource) {
mView = LayoutInflater.from(context).inflate(resource,parent,false);
mView.setTag(this);
} /**
* 获取 CommonViewHold对象,包含 mView和layout里已经添加elements的容器
* @param context
* @param convertView
* @param parent
* @param resource
* @return
*/
public static CommonViewHold getViewHold(Context context ,View convertView,ViewGroup parent,int resource){
if (convertView==null){
return new CommonViewHold(context,parent,resource);
}else{
CommonViewHold viewHold = (CommonViewHold) convertView.getTag();
return viewHold;
}
} public View getView() {
return mView;
} /**
* 维护一个element的容器,如果容器里存在此控件则直接读取,如果不存在则通过findAndSetElement找到元素后
* 写入容器中
* @param viewId
* @param <T>
* @return
*/
public <T extends View> T getElement(int viewId){
View view = elements.get(viewId);
if(view == null){
view = mView.findViewById(viewId);
elements.put(viewId,view);
}
return (T)view;
}
}

通用的adapter

import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter; import java.util.List; /**
* Created by sunfan on 15-4-18.
*/
public abstract class CommonArrayAdapter<T> extends ArrayAdapter{
protected Context mContext;//当前上下文
protected int mResource;//item
protected int mPosition;//当前位置
protected CommonViewHold viewHold;//ViewHold的引用 /**
* 初始化
* @param context
* @param resource
* @param objects
*/
public CommonArrayAdapter(Context context, int resource, List objects) {
super(context, resource, objects);
this.mContext = context;
this.mResource = resource;
} @Override
public View getView(int position, View convertView, ViewGroup parent) {
this.mPosition = position;
viewHold = CommonViewHold.getViewHold(mContext,convertView,parent,mResource);
findAndSetElement();
return viewHold.getView();
} /**
* 需要实现控件的获取与赋值过程
* 这里建议调用内部的viewhold对象里的获取控件的方法
* 例如
* TextView title = super.viewHold.getElement(R.id.title);
title.setText(bean.getTitle());
*/
public abstract void findAndSetElement(); }

客户端

import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.view.Window;
import android.widget.ListView;
import android.widget.TextView; import java.util.ArrayList;
import java.util.List; public class MainActivity extends Activity { ListView listView;
List<Bean> beans = new ArrayList<Bean>(); @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.activity_main);
init();
listView = (ListView) findViewById(R.id.list_view);
listView.setAdapter(new MyAdapter(this, R.layout.list_item, beans)); } /**
* 实现CommonArrayAdapter的findAndSetElement方法
*/
class MyAdapter extends CommonArrayAdapter<Bean> {
public MyAdapter(Context context, int resource, List objects) {
super(context, resource, objects);
} @Override
public void findAndSetElement() {
TextView title = super.viewHold.getElement(R.id.title);
TextView context = super.viewHold.getElement(R.id.context);
Bean bean = beans.get(super.mPosition);
title.setText(bean.getTitle());
context.setText(bean.getContext());
}
} /*
初始化测试数据
*/
private void init() {
Bean b1 = new Bean();
b1.setTitle("title1");
b1.setContext("content1content1content1content1content1content1content1content1content1content1content1content1");
Bean b2 = new Bean();
b2.setTitle("title2");
b2.setContext("content2content2content2content2content2content2content2content2content2");
Bean b3 = new Bean();
b3.setTitle("title3");
b3.setContext("content3content3content3content3content3content3content3content3");
Bean b4 = new Bean();
b4.setTitle("title4");
b4.setContext("content4content4content4content4content4content4content4content4");
Bean b5 = new Bean();
b5.setTitle("title5");
b5.setContext("content5content5content5content5content5content5content5content5content5content5content5");
Bean b6 = new Bean();
b6.setTitle("title6");
b6.setContext("content6content6content6content6content6content6content6content6");
Bean b7 = new Bean();
b7.setTitle("title7");
b7.setContext("content7content7content7content7content7content7content7content7"); beans.add(b1);
beans.add(b2);
beans.add(b3);
beans.add(b4);
beans.add(b5);
beans.add(b6);
beans.add(b7);
}
}

--------------以上为全部代码----------------------

代码分析

整个代码的流程和结构

CommonViewHold 相比传统的Viewhold,他采用容器的方式存储元素,并且封装了客户端用来判断covertview为空的初始化操作

提供了一个存放元素的容器

private SparseArray<View> elements = new SparseArray<View>();//控件元素

当然他提供有获取容器的方法

/**
* 维护一个element的容器,如果容器里存在此控件则直接读取,如果不存在则通过findAndSetElement找到元素后
* 写入容器中
* @param viewId
* @param <T>
* @return
*/
public <T extends View> T getElement(int viewId){
View view = elements.get(viewId);
if(view == null){
view = mView.findViewById(viewId);
elements.put(viewId,view);
}
return (T)view;
}

然后就是做初始化的动作了。

convertView==null的时候的判断,这一段逻辑其实很简单,如果不为空就直接返回,为空就初始化。
private CommonViewHold(Context context, ViewGroup parent, int resource) {
mView = LayoutInflater.from(context).inflate(resource,parent,false);
mView.setTag(this);
} /**
* 获取 CommonViewHold对象,包含 mView和layout里已经添加elements的容器
* @param context
* @param convertView
* @param parent
* @param resource
* @return
*/
public static CommonViewHold getViewHold(Context context ,View convertView,ViewGroup parent,int resource){
if (convertView==null){
return new CommonViewHold(context,parent,resource);
}else{
CommonViewHold viewHold = (CommonViewHold) convertView.getTag();
return viewHold;
}
}

通用的commonAdapter是一个模板类

public abstract class CommonArrayAdapter<T> extends ArrayAdapter

关键属性,当用户实现findAndSet的时候需要使用这个引用来获取控件

  protected  CommonViewHold viewHold;//ViewHold的引用

核心模板方法

 @Override
public View getView(int position, View convertView, ViewGroup parent) {
this.mPosition = position;
viewHold = CommonViewHold.getViewHold(mContext,convertView,parent,mResource);
findAndSetElement();
return viewHold.getView();
}

需要用户去实现的方法

    /**
* 需要实现控件的获取与赋值过程
* 这里建议调用内部的viewhold对象里的获取控件的方法
* 例如
* TextView title = super.viewHold.getElement(R.id.title);
title.setText(bean.getTitle());
*/
public abstract void findAndSetElement();

直接从客户端可以看到,代码是十分便捷的,只需要一个内部类重写commonAdapter的findAndSet方法即可。

 class MyAdapter extends CommonArrayAdapter<Bean> {
public MyAdapter(Context context, int resource, List objects) {
super(context, resource, objects);
} @Override
public void findAndSetElement() {
TextView title = super.viewHold.getElement(R.id.title);
TextView context = super.viewHold.getElement(R.id.context);
Bean bean = beans.get(super.mPosition);
title.setText(bean.getTitle());
context.setText(bean.getContext());
}
}

调用也变得很轻松

    @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.activity_main);
init();
listView = (ListView) findViewById(R.id.list_view);
listView.setAdapter(new MyAdapter(this, R.layout.list_item, beans)); }

android-ViewList的通用ViewHold的更多相关文章

  1. [Cocos2d-x]Android的android.mk文件通用版本

    原文地址: http://blog.ready4go.com/blog/2013/10/12/update-android-dot-mk-with-local-src-files-and-local- ...

  2. iOS,Android,.NET通用AES加密算法

    原文:iOS,Android,.NET通用AES加密算法 这两天为移动App开发API,结果实现加密验证时碰到一大坑.这里不得不吐槽下又臭又硬的iOS,Windows Server无法解密出正确的结果 ...

  3. Android点滴---ViewHolder通用,优雅写法

    近期在做项目时,又要写 ViewHolder. 突然想到网上看看有没什么好的写法! 不知道你是不是也烦透了写那些没有技术含量的ViewHolder 看看这些.也许会有收获! 然后就找到了以下两篇文章( ...

  4. TitleLayout——一个Android轻松实现通用、标准、支持沉浸式状态栏的标题栏库

    TitleLayout 多功能.通用的.可在布局或者使用Java代码实现标题栏:支持沉浸式状态栏,支持左侧返回按钮(不需要手动实现页面返回),左侧支持图片+文字.图片.文字:右侧支持图片.文字等. 堆 ...

  5. Android开发进阶 -- 通用适配器 CommonAdapter

    在Android开发中,我们经常会用到ListView 这个组件,为了将ListView 的内容展示出来,我们会去实现一个Adapter来适配,将Layout中的布局以列表的形式展现到组件中.     ...

  6. Android 异步请求通用类

    package com.example.demo1; import java.util.EventListener; public interface MyAsyncTaskListener exte ...

  7. Android 文件管理器通用类 FileUtil

    1.整体分析 1.1.源代码如下,可以直接Copy. public class FileUtil { private FileUtil() { } //****系统文件目录************** ...

  8. App与Js交互(三)Android、iOS通用解决方案推荐

    https://www.jianshu.com/p/6224f429ce87 window.navigator.userAgent用来区分设备和浏览器 https://blog.csdn.net/li ...

  9. DateTimeUtil 工具类,android 和 java 通用

    import java.sql.Date;import java.text.SimpleDateFormat; public class DateTimeUtil { public final cla ...

随机推荐

  1. 数据库——MySQL——完整性约束

    约束,就是用来保证数据完整性和一致性的. 常见的约束分为: PRIMARY KEY (PK) 标识该字段为该表的主键,可以唯一的标识记录 FOREIGN KEY (FK) 标识该字段为该表的外键 NO ...

  2. react.js中实现tab吸顶效果问题

    在react项目开发中有一个需求是,页面滚动到tab所在位置时,tab要固定在顶部. 实现的思路其实很简单,就是判断当滚动距离scrollTop大于tab距离页面顶部距离offsetTop时,将tab ...

  3. Ajax 跨域的几种解决方案

    作者:黄轩链接:http://www.zhihu.com/question/19618769/answer/38934786来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处 ...

  4. 【学时总结】◆学时·IX◆ 整体二分

    ◆学时·IX◆ 整体二分 至于我怎么了解到这个算法的……只是因为发现一道题,明显的二分查找,但是时间会爆炸,被逼无奈搜题解……然后就发现了一些东西QwQ ◇ 算法概述 整体二分大概是把BFS与二分查找 ...

  5. 【原创】从 列表的重复 到 用sum展开二层嵌套列表将子元素合并

      转载请注明出处:https://www.cnblogs.com/oceanicstar/p/9517159.html     ★像R语言里头有rep函数可以让向量的值重复,在python里面可以直 ...

  6. 【ppp-chap,pap,mp,mp-group】

    PPP链路端口验证(单){ PAP(明文): 主验证方: {local-user user_name:配置本地用户; password {simple||cipher}:配置验证密码; service ...

  7. QP之QEP原理

    1.QP简介: 量子平台(Quantum Platform, 简称QP)是一个用于实时嵌入式系统的软件框架,QP是轻量级的.开源的.基于层次式状态机的.事件驱动的平台. QP包括事件处理器(QEP). ...

  8. u-boot-2016.01移植(一)

    1.了解uboot: 阅读uboot源码顶层目录下的README.TXT可以提取如下信息:     made to support booting of Linux images.   //引导内核程 ...

  9. 基于OMAPL:Linux3.3内核的编译

    基于OMAPL:Linux3.3内核的编译 OMAPL对应3个版本的linux源代码,分别是:Linux-3.3.Linux-2.6.37.Linux2.6.33,这里的差距在于Linux2,缺少SY ...

  10. hive表格取差集

    hive 求两个集合的差集 业务场景是这样的,这里由两个hive表格A和B A的形式大概是这样的:uid B的形式大概是这样的:uid 我想要得到存在A中但是不存在B中的uid 具体代码如下 sele ...