RecyclerView实现多type页面

01.先看看实际需求


比如一个APP的首页,包含Banner区、广告区、文本内容、图片内容、新闻内容等等。

RecyclerView 可以用ViewType来区分不同的item,也可以满足需求,但还是存在一些问题,比如:

在item过多逻辑复杂列表界面,Adapter里面的代码量庞大,逻辑复杂,后期难以维护。

每次增加一个列表都需要增加一个Adapter,重复搬砖,效率低下。

无法复用adapter,假如有多个页面有多个type,那么就要写多个adapter。

要是有局部刷新,那么就比较麻烦了,比如广告区也是一个九宫格的RecyclerView,点击局部刷新当前数据,比较麻烦。

02.adapter实现多个type


通常写一个多Item列表的方法

  • 根据不同的ViewType

处理不同的item,如果逻辑复杂,这个类的代码量是很庞大的。如果版本迭代添加新的需求,修改代码很麻烦,后期维护困难。

主要操作步骤

  • 在onCreateViewHolder中根据viewType参数,也就是getItemViewType的返回值来判断需要创建的ViewHolder类型

  • 在onBindViewHolder方法中对ViewHolder的具体类型进行判断,分别为不同类型的ViewHolder进行绑定数据与逻辑处理

代码如下所示

public class HomePageAdapter extends RecyclerView.Adapter {

public static final int TYPE_BANNER = 0;

public static final int TYPE_AD = 1;

public static final int TYPE_TEXT = 2;

public static final int TYPE_IMAGE = 3;

public static final int TYPE_NEW = 4;

private List mData;

public void setData(List data) {

mData = data;

}

@Override

public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

switch (viewType){

case TYPE_BANNER:

return new BannerViewHolder(LayoutInflater.from(parent.getContext()).inflate(R.layout.home_banner_layout,null));

case TYPE_AD:

return new BannerViewHolder(LayoutInflater.from(parent.getContext()).inflate(R.layout.home_ad_item_layout,null));

case TYPE_TEXT:

return new BannerViewHolder(LayoutInflater.from(parent.getContext()).inflate(R.layout.home_text_item_layout,null));

case TYPE_IMAGE:

return new BannerViewHolder(LayoutInflater.from(parent.getContext()).inflate(R.layout.home_image_item_layout,null));

case TYPE_NEW:

return new BannerViewHolder(LayoutInflater.from(parent.getContext()).inflate(R.layout.home_news_item_layout,null));

}

return null;

}

@Override

public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {

int type = getItemViewType(position);

switch (type){

case TYPE_BANNER:

// banner 逻辑处理

break;

case TYPE_AD:

// 广告逻辑处理

break;

case TYPE_TEXT:

// 文本逻辑处理

break;

case TYPE_IMAGE:

//图片逻辑处理

break;

case TYPE_NEW:

//视频逻辑处理

break;

// … 此处省去N行代码

}

}

@Override

public int getItemViewType(int position) {

if(position == 0){

return TYPE_BANNER;//banner在开头

}else {

return mData.get(position).type;//type 的值为TYPE_AD,TYPE_IMAGE,TYPE_AD,等其中一个

}

}

@Override

public int getItemCount() {

return mData == null ? 0:mData.size();

}

public static class BannerViewHolder extends RecyclerView.ViewHolder{

public BannerViewHolder(View itemView) {

super(itemView);

//绑定控件

}

}

public static class NewViewHolder extends RecyclerView.ViewHolder{

public VideoViewHolder(View itemView) {

super(itemView);

//绑定控件

}

}

public static class AdViewHolder extends RecyclerView.ViewHolder{

public AdViewHolder(View itemView) {

super(itemView);

//绑定控件

}

}

public static class TextViewHolder extends RecyclerView.ViewHolder{

public TextViewHolder(View itemView) {

super(itemView);

//绑定控件

}

}

public static class ImageViewHolder extends RecyclerView.ViewHolder{

public ImageViewHolder(View itemView) {

super(itemView);

//绑定控件

}

}

}

03.这样写的弊端


上面那样写的弊端

  • 类型检查与类型转型,由于在onCreateViewHolder根据不同类型创建了不同的ViewHolder,所以在onBindViewHolder需要针对不同类型的ViewHolder进行数据绑定与逻辑处理,这导致需要通过instanceof对ViewHolder进行类型检查与类型转型。

  • 不利于扩展,目前的需求是列表中存在5种布局类类型,那么如果需求变动,极端一点的情况就是数据源是从服务器获取的,数据中的model决定列表中的布局类型。这种情况下,每当model改变或model类型增加,我们都要去改变adapter中很多的代码,同时Adapter还必须知道特定的model在列表中的位置(position)除非跟服务端约定好,model(位置)不变,很显然,这是不现实的。

  • 不利于维护,这点应该是上一点的延伸,随着列表中布局类型的增加与变更,getItemViewType、onCreateViewHolder、onBindViewHolder中的代码都需要变更或增加,Adapter

中的代码会变得臃肿与混乱,增加了代码的维护成本。

04.如何优雅实现adapter封装


核心目的就是三个

  • 避免类的类型检查与类型转型

  • 增强Adapter的扩展性

  • 增强Adapter的可维护性

当列表中类型增加或减少时Adapter中主要改动的就是getItemViewType、onCreateViewHolder、onBindViewHolder这三个方法,因此,我们就从这三个方法中开始着手。

既然可能存在多个type类型的view,那么能不能把这些比如banner,广告,文本,视频,新闻等当做一个HeaderView来操作。

在getItemViewType方法中。

  • 减少if之类的逻辑判断简化代码,可以简单粗暴的用hashCode作为增加type标识。

  • 通过创建列表的布局类型,同时返回的不再是简单的布局类型标识,而是布局的hashCode值

private ArrayList headers = new ArrayList<>();

public interface InterItemView {

/**

  • 创建view

  • @param parent parent

  • @return view

*/

View onCreateView(ViewGroup parent);

/**

  • 绑定view

  • @param headerView headerView

*/

void onBindView(View headerView);

}

/**

  • 获取类型,主要作用是用来获取当前项Item(position参数)是哪种类型的布局

  • @param position 索引

  • @return int

*/

@Deprecated

@Override

public final int getItemViewType(int position) {

if (headers.size()!=0){

if (position<headers.size()) {

return headers.get(position).hashCode();

}

}

if (footers.size()!=0){

int i = position - headers.size() - mObjects.size();

if (i >= 0){

return footers.get(i).hashCode();

}

}

return getViewType(position-headers.size());

}

onCreateViewHolder

  • getItemViewType返回的是布局hashCode值,也就是onCreateViewHolder(ViewGroup

parent, int viewType)参数中的viewType

/**

  • 创建viewHolder,主要作用是创建Item视图,并返回相应的ViewHolder

  • @param parent parent

  • @param viewType type类型

  • @return 返回viewHolder

*/

@NonNull

@Override

public final BaseViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {

View view = createViewByType(parent, viewType);

if (view!=null){

return new BaseViewHolder(view);

最后

最后这里放上我这段时间复习的资料,这个资料也是偶然一位朋友分享给我的,里面包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。

还有 高级架构技术进阶脑图、高级进阶架构资料 帮助大家学习提升进阶,也可以分享给身边好友一起学习。

一起互勉~
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
eViewByType(parent, viewType);

if (view!=null){

return new BaseViewHolder(view);

最后

最后这里放上我这段时间复习的资料,这个资料也是偶然一位朋友分享给我的,里面包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。

还有 高级架构技术进阶脑图、高级进阶架构资料 帮助大家学习提升进阶,也可以分享给身边好友一起学习。

[外链图片转存中…(img-mg64CBfx-1715866869373)]

[外链图片转存中…(img-IvgHCmRv-1715866869374)]

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

  • 30
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
首先,我们需要在AndroidManifest.xml文件中添加文件读写权限: ``` <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> ``` 接着,我们需要创建一个广播接收器来监听系统数据库变化: ``` public class MediaReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (action != null && action.equals(Intent.ACTION_MEDIA_SCANNER_FINISHED)) { // 扫描完成,可以开始获取媒体文件 List<MediaFile> mediaFiles = getMediaFiles(context); // 将媒体文件展示在RecyclerView中 updateRecyclerView(mediaFiles); } } private List<MediaFile> getMediaFiles(Context context) { List<MediaFile> mediaFiles = new ArrayList<>(); ContentResolver contentResolver = context.getContentResolver(); // 查询媒体库中的视频文件 Cursor cursor = contentResolver.query(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, null, null, null, null); if (cursor != null) { while (cursor.moveToNext()) { int id = cursor.getInt(cursor.getColumnIndex(MediaStore.Video.Media._ID)); String title = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.TITLE)); String path = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.DATA)); long duration = cursor.getLong(cursor.getColumnIndex(MediaStore.Video.Media.DURATION)); long size = cursor.getLong(cursor.getColumnIndex(MediaStore.Video.Media.SIZE)); String mimeType = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.MIME_TYPE)); // 判断文件是否存在 File file = new File(path); if (file.exists()) { // 将媒体文件添加到列表中 MediaFile mediaFile = new MediaFile(id, title, path, duration, size, mimeType); mediaFiles.add(mediaFile); } } cursor.close(); } return mediaFiles; } private void updateRecyclerView(List<MediaFile> mediaFiles) { // 更新RecyclerView中的数据 // ... } } ``` 接下来,在Activity中注册广播接收器: ``` private MediaReceiver mediaReceiver; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 注册广播接收器 mediaReceiver = new MediaReceiver(); IntentFilter intentFilter = new IntentFilter(); intentFilter.addAction(Intent.ACTION_MEDIA_SCANNER_FINISHED); registerReceiver(mediaReceiver, intentFilter); } @Override protected void onDestroy() { super.onDestroy(); // 注销广播接收器 unregisterReceiver(mediaReceiver); } ``` 最后,在RecyclerView的Adapter中更新数据: ``` public class MediaAdapter extends RecyclerView.Adapter<MediaAdapter.ViewHolder> { private List<MediaFile> mediaFiles; public MediaAdapter(List<MediaFile> mediaFiles) { this.mediaFiles = mediaFiles; } @Override public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_media, parent, false); return new ViewHolder(view); } @Override public void onBindViewHolder(ViewHolder holder, int position) { MediaFile mediaFile = mediaFiles.get(position); holder.titleTextView.setText(mediaFile.getTitle()); holder.durationTextView.setText(formatDuration(mediaFile.getDuration())); } @Override public int getItemCount() { return mediaFiles.size(); } public void updateData(List<MediaFile> mediaFiles) { this.mediaFiles = mediaFiles; notifyDataSetChanged(); } public static class ViewHolder extends RecyclerView.ViewHolder { private TextView titleTextView; private TextView durationTextView; public ViewHolder(View itemView) { super(itemView); titleTextView = itemView.findViewById(R.id.titleTextView); durationTextView = itemView.findViewById(R.id.durationTextView); } } private String formatDuration(long duration) { long minutes = TimeUnit.MILLISECONDS.toMinutes(duration); long seconds = TimeUnit.MILLISECONDS.toSeconds(duration) - TimeUnit.MINUTES.toSeconds(minutes); return String.format(Locale.getDefault(), "%d:%02d", minutes, seconds); } } ``` 这样,每次新增视频文件时,广播接收器会接收到系统数据库变化的通知,然后获取新增的媒体文件并展示在RecyclerView中。如果跳转到其他页面再回来,也会重新获取媒体文件并展示在RecyclerView中。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值