0

0

如何在 RecyclerView 中精准更新指定位置的图片(下载完成后)

花韻仙語

花韻仙語

发布时间:2025-12-31 18:46:23

|

394人浏览过

|

来源于php中文网

原创

如何在 RecyclerView 中精准更新指定位置的图片(下载完成后)

本文详解如何在图片异步下载完成(onsuccess)后,安全、高效地仅刷新 recyclerview 中对应 position 的 viewholder,避免 notifydatasetchanged() 导致的闪烁、缩放错乱与性能损耗,并提供线程安全、缓存友好、可维护的实践方案。

在 Android 开发中,使用 RecyclerView 展示网络图片时,一个常见且关键的需求是:图片下载成功后,只更新当前 item 的 ImageView,而非整个列表。你遇到的问题——notifyItemChanged(position) 在 onSuccess() 中无效——根本原因在于:该回调发生在子线程(ImageLoader 的 ExecutorService 线程池中),而 RecyclerView.Adapter 的所有 notify 方法必须在主线程(UI 线程)调用

直接调用 notifyItemChanged(position) 会导致无响应或崩溃(尤其在 Debug 模式下启用 StrictMode 时会抛出 CalledFromWrongThreadException)。而滥用 notifyDataSetChanged() 虽能“生效”,却会重置所有 ViewHolder 的状态(如 ImageView.ScaleType、滚动位置、动画等),造成视觉跳变和体验劣化。

✅ 正确做法:确保 UI 更新在主线程执行

将 notifyItemChanged(position) 包裹在主线程调度中即可解决:

@Override
public void onSuccess(Bitmap bitmap) {
    // ✅ 安全:确保在主线程更新 UI 和 Adapter 状态
    holder.itemView.post(() -> {
        holder.comic_page.setImageBitmap(bitmap);
        holder.comic_page.setScaleType(ImageView.ScaleType.CENTER_INSIDE);
        notifyItemChanged(position); // ✅ 此时已在主线程
    });
}
? 推荐使用 holder.itemView.post(Runnable) 而非 runOnUiThread(),因为它更轻量、无需持有 Activity 引用,避免潜在内存泄漏,且天然绑定到当前 ViewHolder 的生命周期。

? 进阶优化:避免重复绑定与竞态条件

当前代码存在两个隐患,需一并修复:

富兰氏手机商城系统源码 2011
富兰氏手机商城系统源码 2011

全国首个为手机行业定制的网站,外观豪华、时尚。DIV+CSS构建,符合W3C标准,完美搜索引擎优化迅速提高搜索引擎排名,稳定性、执行效率、负载能力均居国内同类产品领先地位。安装简单,傻瓜式操作,在线下单、支付、发货,轻松管理网站。 多套模板更换,界面更加豪华 完美搜索引擎优化 集成支付宝、财付通、网银等多种在线支付平台 手机、配件商品不同颜色、型号不同价格设置 图片化多种参数设置、搜索、评论 新闻

下载
  1. 重复加载风险:onBindViewHolder() 可能被多次调用(如滑动复用、列表刷新),而 ImageLoader.load(...) 未做 view 复用校验,可能导致旧请求结果错误覆盖新请求;
  2. Bitmap 冗余设置:onSuccess 中已由 Displacer 设置了 imageView.setImageBitmap(bitmap),你在回调里又设了一次,属于冗余操作。

推荐重构 onBindViewHolder 如下

@Override
public void onBindViewHolder(@NonNull ComicViewHolder holder, int position) {
    String imageUrl = Manga.resolveURL(linksList.get(position), context);

    // ✅ 关键:清除可能存在的旧请求关联(防复用错位)
    Manga.imageLoader.cancelRequest(holder.comic_page);

    Manga.imageLoader.with(context, context.getCacheDir() + "/cache")
            .load(imageUrl, holder.comic_page, new LoadImage() {
                @Override
                public void onSuccess(Bitmap bitmap) {
                    // ✅ 仅在主线程安全更新
                    holder.itemView.post(() -> {
                        // 确保 ImageView 仍绑定此 position(防快速滑动导致的复用错位)
                        if (holder.getAdapterPosition() == position && !holder.isRemoved()) {
                            holder.comic_page.setScaleType(ImageView.ScaleType.CENTER_INSIDE);
                            // Bitmap 已由 ImageLoader 自动设置,此处无需再 set
                            notifyItemChanged(position); // 触发重新绑定,确保状态一致
                        }
                    });
                }

                @Override
                public void onFail() {
                    holder.itemView.post(() -> {
                        if (holder.getAdapterPosition() == position && !holder.isRemoved()) {
                            // 可选:显示占位图或错误图标
                            holder.comic_page.setImageResource(R.drawable.ic_error);
                            notifyItemChanged(position);
                        }
                    });
                }
            }, null);
}

同时,强烈建议为 ImageLoader 补充 cancelRequest(ImageView) 方法(基于 imageViews Map 实现),用于解绑即将被复用的 ImageView,防止回调执行时 view 已指向其他数据:

// 在 ImageLoader 类中添加
public void cancelRequest(ImageView imageView) {
    String url = imageViews.remove(imageView);
    if (url != null) {
        // 可选:取消对应线程任务(需增强 PhotoLoader 的可取消性)
    }
}

? 注意事项与最佳实践

  • *永远不要在子线程直接调用 `notify` 方法**:这是 RecyclerView 的硬性约束;
  • 始终校验 ViewHolder 状态:使用 holder.getAdapterPosition() 和 !holder.isRemoved() 防止因异步延迟导致的 UI 错位;
  • 避免过度 notify:notifyItemChanged(position) 会触发 onBindViewHolder() 重入,若你的 onBindViewHolder 中逻辑复杂(如再次发起网络请求),需加锁或标记位控制;
  • 考虑现代替代方案:ImageLoader 手动实现较重,推荐迁移到成熟库如 GlideCoil,它们内置线程切换、自动请求管理、生命周期感知,一行代码即可安全加载:
    Glide.with(holder.itemView.context)
         .load(imageUrl)
         .centerInside()
         .into(holder.comic_page)

✅ 总结

精准更新 RecyclerView 单个 item 图片的核心就三点:
线程安全——notifyItemChanged() 必须在主线程;
状态校验——确认 ViewHolder 仍有效且 position 未变;
请求解耦——及时取消复用 View 的旧加载任务。

按上述方式改造后,你将获得丝滑、稳定、高性能的图片加载体验,彻底告别 notifyDataSetChanged() 带来的副作用。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

471

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

471

2023.08.10

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

73

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

25

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

36

2025.11.17

java判断map相关教程
java判断map相关教程

本专题整合了java判断map相关教程,阅读专题下面的文章了解更多详细内容。

31

2025.11.27

CSS position定位有几种方式
CSS position定位有几种方式

有4种,分别是静态定位、相对定位、绝对定位和固定定位。更多关于CSS position定位有几种方式的内容,可以访问下面的文章。

80

2023.11.23

android开发三大框架
android开发三大框架

android开发三大框架是XUtil框架、volley框架、ImageLoader框架。本专题为大家提供android开发三大框架相关的各种文章、以及下载和课程。

249

2023.08.14

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Excel 教程
Excel 教程

共162课时 | 10.1万人学习

Java 教程
Java 教程

共578课时 | 39.9万人学习

Uniapp从零开始实现新闻资讯应用
Uniapp从零开始实现新闻资讯应用

共64课时 | 6.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号