[Android] Android RecycleView和ListView 自定义Adapter封装类
在网上查看了很多对应 Android RecycleView和ListView 自定义Adapter封装类 的文章,主要存在几个问题:
一)、网上代码一大抄,复制来复制去,大部分都运行不起来,或者 格式错乱
二)、剩下的那些能运行起来的,将Adapter类、ViewHolder类,放在不同文件里,导致文件多、杂
于是总结这两个单独的Custom***ViewAdapter 类,以方便调用!
一、RecycleView Adapter自定义封装类
CustomRecyclerViewAdapter.java
package com.jack.androidbase.adapter; import android.content.Context;
import android.support.v7.widget.RecyclerView;
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 java.util.List; public abstract class CustomRecyclerViewAdapter<T> extends RecyclerView.Adapter<CustomRecyclerViewAdapter.ViewHolder> {
private int mLayoutRes;
private List<T> mData; //定义接口 OnItemClickListener
public interface OnItemClickListener {
void onItemClick(View view, int position); void onItemLongClick(View view, int position);
} private OnItemClickListener mOnItemClickListener; public void setOnItemClickListener(OnItemClickListener onItemClickListener) {
mOnItemClickListener = onItemClickListener;
} public CustomRecyclerViewAdapter(List<T> mData, int mLayoutRes) {
this.mLayoutRes = mLayoutRes;
this.mData = mData;
} /**
* inflate操作
*/
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
ViewHolder viewHolder = ViewHolder.get(parent.getContext(), parent, mLayoutRes);
return viewHolder;
} /**
* 暴露出来的接口
*
* @param holder
* @param obj
*/
public abstract void bindView(ViewHolder holder, T obj); @Override
public void onBindViewHolder(final ViewHolder holder, int position) {
bindView(holder, mData.get(position)); //绑定事件
if (mOnItemClickListener != null) {
holder.itemView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
int pos = holder.getLayoutPosition();
mOnItemClickListener.onItemClick(holder.itemView, pos);
}
});
holder.itemView.setOnLongClickListener(new View.OnLongClickListener() {
@Override
public boolean onLongClick(View v) {
int pos = holder.getLayoutPosition();
mOnItemClickListener.onItemLongClick(holder.itemView, pos);
return false;
}
});
}
} /**
* 获取高度
*
* @return
*/
@Override
public int getItemCount() {
if (mData == null) {
return 0;
}
return mData.size();
} /**
* ViewHolder
*/
public static class ViewHolder extends RecyclerView.ViewHolder {
private SparseArray<View> mViews;
private View mConvertView; public ViewHolder(Context context, View itemView, ViewGroup parent) {
super(itemView);
mConvertView = itemView;
mViews = new SparseArray<View>();
} public static ViewHolder get(Context context, ViewGroup parent, int layoutid) {
View itemView = LayoutInflater.from(context).inflate(layoutid, parent, false);
ViewHolder holder = new ViewHolder(context, itemView, parent);
return holder;
} 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文本
*/
public ViewHolder setText(int viewId, CharSequence text) {
View view = getView(viewId);
if (view instanceof TextView) {
((TextView) view).setText(text);
}
return this;
} /**
* 设置View的Visibility
*/
public ViewHolder setViewVisibility(int viewId, int visibility) {
getView(viewId).setVisibility(visibility);
return this;
} /**
* 设置ImageView的资源
*/
public ViewHolder setImageResource(int viewId, int drawableRes) {
View view = getView(viewId);
if (view instanceof ImageView) {
((ImageView) view).setImageResource(drawableRes);
} else {
view.setBackgroundResource(drawableRes);
}
return this;
}
} }
如何使用:
MainActivity.java 调用
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_ui_recycler_view2); bindView();
} private void bindView() {
mRecyclerView = (RecyclerView) findViewById(R.id.ui_basic_recycler_custom_1); mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
mData = new ArrayList();
for (int i = 0; i < 10; i++) {
mData.add("" + i);
} //数据装载
myAdapter = new CustomRecyclerViewAdapter(mData, R.layout.item_recyler_1) {
@Override
public void bindView(ViewHolder holder, Object obj) {
holder.setText(R.id.recycler_txt_1, obj.toString());
}
}; //点击监听
listener1 = new CustomRecyclerViewAdapter.OnItemClickListener(){
@Override
public void onItemClick(View view, int position) {
Toast.makeText(getApplicationContext(), position + " click", Toast.LENGTH_SHORT).show();
} @Override
public void onItemLongClick(View view, int position) {
Toast.makeText(getApplicationContext(), position + " Long click", Toast.LENGTH_SHORT).show();
}
};
mRecyclerView.setAdapter(myAdapter);
myAdapter.setOnItemClickListener(listener1);
}
activity_ui_recycler_view2.xml 布局文件
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".activity.***Activity"> <android.support.v7.widget.RecyclerView
android:id="@+id/ui_basic_recycler_custom_1"
android:layout_width="match_parent"
android:layout_height="wrap_content"></android.support.v7.widget.RecyclerView>
</RelativeLayout>
二、ListView Adapter 自定义封装类
CustomListViewAdapter.java
package com.jack.androidbase.adapter; 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.ImageView;
import android.widget.TextView; import com.jack.androidbase.R; import java.util.ArrayList;
import java.util.LinkedList; public abstract class CustomListViewAdapter<T> extends BaseAdapter { private ArrayList<T> mData;
private int mLayoutRes; //布局id public CustomListViewAdapter(ArrayList<T> mData, int mLayoutRes) {
this.mData = mData;
this.mLayoutRes = mLayoutRes;
} @Override
public int getCount() {
return mData != null ? mData.size() : 0;
} @Override
public T getItem(int position) {
return mData.get(position);
} @Override
public long getItemId(int position) {
return position;
} @Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = ViewHolder.bind(parent.getContext(), convertView, parent, mLayoutRes
, position);
bindView(holder, getItem(position));
return holder.getItemView();
} public abstract void bindView(ViewHolder holder, T obj); //添加一个元素
public void add(T data) {
if (mData == null) {
mData = new ArrayList<>();
}
mData.add(data);
notifyDataSetChanged();
} //往特定位置,添加一个元素
public void add(int position, T data) {
if (mData == null) {
mData = new ArrayList<>();
}
mData.add(position, data);
notifyDataSetChanged();
} public void remove(T data) {
if (mData != null) {
mData.remove(data);
}
notifyDataSetChanged();
} public void remove(int position) {
if (mData != null) {
mData.remove(position);
}
notifyDataSetChanged();
} public void clear() {
if (mData != null) {
mData.clear();
}
notifyDataSetChanged();
} public static class ViewHolder { private SparseArray<View> mViews; //存储ListView 的 item中的View
private View item; //存放convertView
private int position; //游标
private Context context; //Context上下文 //构造方法,完成相关初始化
private ViewHolder(Context context, ViewGroup parent, int layoutRes) {
mViews = new SparseArray<>();
this.context = context;
View convertView = LayoutInflater.from(context).inflate(layoutRes, parent, false);
convertView.setTag(this);
item = convertView;
} //绑定ViewHolder与item
public static ViewHolder bind(Context context, View convertView, ViewGroup parent,
int layoutRes, int position) {
ViewHolder holder;
if (convertView == null) {
holder = new ViewHolder(context, parent, layoutRes);
} else {
holder = (ViewHolder) convertView.getTag();
holder.item = convertView;
}
holder.position = position;
return holder;
} @SuppressWarnings("unchecked")
public <T extends View> T getView(int id) {
T t = (T) mViews.get(id);
if (t == null) {
t = (T) item.findViewById(id);
mViews.put(id, t);
}
return t;
} /**
* 获取当前条目
*/
public View getItemView() {
return item;
} /**
* 获取条目位置
*/
public int getItemPosition() {
return position;
} /**
* 设置文字
*/
public ViewHolder setText(int id, CharSequence text) {
View view = getView(id);
if (view instanceof TextView) {
((TextView) view).setText(text);
}
return this;
} /**
* 设置图片
*/
public ViewHolder setImageResource(int id, int drawableRes) {
View view = getView(id);
if (view instanceof ImageView) {
((ImageView) view).setImageResource(drawableRes);
} else {
view.setBackgroundResource(drawableRes);
}
return this;
} /**
* 设置点击监听
*/
public ViewHolder setOnClickListener(int id, View.OnClickListener listener) {
getView(id).setOnClickListener(listener);
return this;
} /**
* 设置可见
*/
public ViewHolder setVisibility(int id, int visible) {
getView(id).setVisibility(visible);
return this;
} /**
* 设置标签
*/
public ViewHolder setTag(int id, Object obj) {
getView(id).setTag(obj);
return this;
} //其他方法可自行扩展 } }
如何使用:
MainActivity.java 调用
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_ui_base_list_view_custom); mContext = UIBaseListViewCustomActivity.this;
initList();
} private void initList() {
list_book = (ListView) findViewById(R.id.ui_base_list_v2_1); //数据初始化
mData1 = new ArrayList<>();
mData1.add(new Animal("狗说", "你是狗么?", R.drawable.avatar_1));
mData1.add(new Animal("牛说", "你是牛么?", R.drawable.avatar_2));
mData1.add(new Animal("鸭说", "你是鸭么?", R.drawable.avatar_3)); //Adapter初始化
myAdapter1 = new CustomListViewAdapter<Animal>((ArrayList) mData1, R.layout.item_list_animal) {
@Override
public void bindView(ViewHolder holder, Animal obj) {
holder.setImageResource(R.id.img_icon, obj.getaIcon());
holder.setText(R.id.txt_aName, obj.getaName());
}
}; //ListView设置下Adapter:
list_book.setAdapter(myAdapter1); list_book.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Toast.makeText(mContext, "你点击了~" + position + "~项", Toast.LENGTH_SHORT).show();
}
});
}
item_list_animal 布局文件 :
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".activity.UIBaseListViewActivity"> <ImageView
android:id="@+id/img_icon"
android:layout_width="64dp"
android:layout_height="64dp"
android:baselineAlignBottom="true"
android:paddingLeft="8dp" /> <LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:paddingLeft="20dp"
android:paddingTop="10dp"> <TextView
android:id="@+id/txt_aName"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#1D1D1C"
android:textSize="20sp" /> <TextView
android:id="@+id/txt_aSpeak"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="#B4B4B9"
android:textSize="14sp" /> </LinearLayout> </LinearLayout>
三、Github仓库代码:
https://github.com/wukong1688/AndroidBase/tree/master/app/src/main/java/com/jack/androidbase/adapter
如果对您有帮助,欢迎 点赞,欢迎Star,并转发!
本博客地址: wukong1688
本文原文地址: https://www.cnblogs.com/wukong1688/p/10661391.html
转载请著名出处!谢谢~~
[Android] Android RecycleView和ListView 自定义Adapter封装类的更多相关文章
- Android ListView 自定义 Adapter
自定义Adapter类 public class ListViewAdapter extends BaseAdapter { private static final String TAG = Mai ...
- Android中利用ViewHolder优化自定义Adapter的典型写法
利用ViewHolder优化自定义Adapter的典型写法 最近写Adapter写得多了,慢慢就熟悉了. 用ViewHolder,主要是进行一些性能优化,减少一些不必要的重复操作.(WXD同学教我的. ...
- Xamarin.Android 入门之:Listview和adapter
一.引言 不管开发什么软件,列表的使用是必不可少的,而本章我们将学习如何使用Xamarin去实现它,以及如何使用自定义适配器.关于xamarin中listview的基础和适配器可以查看官网https: ...
- Android修行之路------ListView自定义布局
主布局 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android= ...
- Android ListView自定义Adapter使用误区
参考博客:BaseAdapter中重写getview的心得以及发现convertView回收的机制 使用自定义的BaseAdapter实现LIstView的展示 由于Recycler(反复循环器)的机 ...
- 【Android】自己定义ListView的Adapter报空指针异常解决方法
刚刚使用ViewHolder的方法拉取ListView的数据,可是总会报异常. 细致查看代码.都正确. 后来打开adapter类,发现getView的返回值为null. 即return null. 将 ...
- Android开发之关于ListView中adapter调用notifyDataSetChanged无效的原因
1.数据源没有更新,调用notifyDataSetChanged无效. 2.数据源更新了,但是它指向新的引用,调用notifyDataSetChanged无效. 3.数据源更新了,但是adpter没有 ...
- Android开发系列之ListView
上篇博客解决了Android客户端通过WebService与服务器端程序进行交互的问题,这篇博客重点关注两个问题,一个是Android应用程序如何与本机文件型数据库SQLite进行交互,另一问题则是如 ...
- Android控件开发——ListView
上篇博客解决了Android客户端通过WebService与服务器端程序进行交互的问题,这篇博客重点关注两个问题,一个是Android应用程序如何与本机文件型数据库SQLite进行交互,另一问题则是如 ...
随机推荐
- 【题解】 bzoj3555: [Ctsc2014]企鹅QQ (字符串Hash)
题面戳我 Solution 我们分析题意,他要求的是两个字符串只有一个字符不同,然后我们再看长度\(L \leq 200\),显然我们就可以把每一位删除后\(Hash\),然后判断相同个数即可 我一开 ...
- (1)Phonics自然拼读 英语动画 Fun with Phonics 国际主流英语教学法
Phonics(拼音英语)是目前国际主流的英语教学法,我国香港和台湾地区2000年就已引进此教学法,并已进入大规模推广和普及阶段.它之所以风靡全球,是因为这种教学法简单高效,符合人类学习语言的规律,尤 ...
- redis 指令文档
参考:https://redis.io/commands http://www.runoob.com/redis/redis-lists.html redis: 打开一个 cmd 窗口 使用cd命令切 ...
- js jquery 数组的合并 对象的合并
转载自:http://www.cnblogs.com/xingxiangyi/p/6416468.html 1 数组合并 1.1 concat 方法 1 2 3 4 var a=[1,2,3],b=[ ...
- Manjaro更新出现冲突
➜ ~ sudo pacman -Syyu :: Synchronizing package databases... core 147.5 KiB 1378K/s 00:00 [########## ...
- 在views中引用UserProfile报错RuntimeError: Model class apps.users.models.UserProfile doesn't declare an explicit app_label and isn't in an application in INSTALLED_APPS.
如图报错,在settings中,该加的也加了啊! 显然类似于网上最容易遇到的解决方案如下图,是没有任何意义的 只要在view中有 from .models import UserProfile,Ver ...
- CF954F Runner's Problem(DP+矩阵快速幂优化)
这题是一年前某场我参加过的Education Round codeforces的F题,当时我显然是不会的. 现在看看感觉应该是能做出的. 不扯了写题解: 考虑朴素的DP,在不存在障碍的情况下:f[i] ...
- 第三篇-以LinearLayout进行Android界面设计
一.新建一个项目 选择empty activity,此时项目里面没有main.java的文件. 二.手动创建java文件 Project那儿选择android模式,在app/java/com....一 ...
- Redis高并发和快速的原因
一.Redis的高并发和快速原因 1.redis是基于内存的,内存的读写速度非常快: 2.redis是单线程的,省去了很多上下文切换线程的时间: 3.redis使用多路复用技术,可以处理并发的连接 ...
- TestNg1. 基本介绍注解介绍和如何让在maven中引用
1.更适合测试人员,有很多的套件. maven中引用: <!-- https://mvnrepository.com/artifact/org.testng/testng --><d ...