0

0

如何在嵌套 RecyclerView 中准确传递并汇总多层商品价格

聖光之護

聖光之護

发布时间:2026-02-04 18:12:08

|

109人浏览过

|

来源于php中文网

原创

如何在嵌套 RecyclerView 中准确传递并汇总多层商品价格

本文详解如何在三层嵌套 recyclerview(商店 → 商品 → 附加项)结构中,正确计算并向上回传每家店铺的总价(含商品主价、附加项价格及数量),避免因 adapter 生命周期与数据绑定时机导致的数值丢失问题。

在 Android 开发中,使用嵌套 RecyclerView 实现复杂层级列表(如购物车按商家分组)虽灵活,但极易陷入“状态不可靠”陷阱——尤其是涉及跨层级数据聚合(如总价计算)时。你当前遇到的核心问题:CartItemAdapter 内部能正确累加 totalItems 并输出日志,但在 CartStoreAdapter 中调用 getTotalItems() 却返回 0.0,根本原因在于 totalItems 是一个未初始化、未持久化、且未在 Adapter 全生命周期内被可靠维护的临时变量

为什么 totalItems += subTotal 在 ViewHolder 中失效?

  • CartItemAdapter 的 onBindViewHolder() 会被反复调用(复用机制),每次绑定新 item 时,totalItems 都可能被覆盖或重置;
  • getTotalItems() 被调用时(在 CartStoreAdapter.ItemRowHolder.bind() 中),CartItemAdapter 尚未完成全部 onBindViewHolder() 执行,totalItems 仍为初始值 0.0;
  • 更关键的是:totalItems 属于 Adapter 实例状态,但其更新逻辑分散在 ViewHolder 的 bind() 中,缺乏同步保障,也违背了「数据驱动 UI」的设计原则。

✅ 推荐方案:预计算 + 不可变数据模型

最佳实践是将总价计算逻辑前置到数据层,而非依赖 UI 组件(Adapter/ViewHolder)临时聚合。 这样既保证数据一致性,又解耦 UI 与业务逻辑,提升可测试性与性能。

魔术橡皮擦
魔术橡皮擦

智能擦除、填补背景内容

下载

步骤一:增强 CartStoreModel 模型,支持总价缓存

public class CartStoreModel {
    private String nama_merchant;
    private List cartItem;
    private double totalStorePrice; // ← 新增字段:该店总金额(含商品+附加项+数量)

    // 构造时即计算,或提供计算方法
    public void calculateTotalStorePrice() {
        this.totalStorePrice = 0.0;
        if (cartItem != null) {
            for (CartItemModel item : cartItem) {
                double itemBasePrice = item.getHarga_promo() > 0 ? item.getHarga_promo() : item.getHarga_item();
                double addonPrice = item.getCartItemVar() != null 
                    ? item.getCartItemVar().stream()
                        .mapToDouble(CartItemVarModel::getVar_price)
                        .sum() 
                    : 0.0;
                double subTotal = (itemBasePrice + addonPrice) * item.getQty();
                this.totalStorePrice += subTotal;
            }
        }
    }

    // Getter
    public double getTotalStorePrice() {
        return totalStorePrice;
    }
}

步骤二:在设置 Adapter 前完成预计算(推荐在 ViewModel 或 Repository 层)

// StoreViewModel.java
public class StoreViewModel extends ViewModel {
    private MutableLiveData> storeListLiveData = new MutableLiveData<>();

    public void loadCartStores(List rawStores) {
        // ✅ 在后台线程计算总价(避免主线程阻塞)
        new Thread(() -> {
            for (CartStoreModel store : rawStores) {
                store.calculateTotalStorePrice(); // 触发预计算
            }
            // 切回主线程更新 UI
            postValue(rawStores);
        }).start();
    }

    private void postValue(List value) {
        storeListLiveData.postValue(value);
    }

    public LiveData> getStoreListLiveData() {
        return storeListLiveData;
    }
}

步骤三:CartStoreAdapter 直接读取已计算好的总价

public class CartStoreAdapter extends RecyclerView.Adapter {
    private List storeDataList;
    private Context mContext;
    private int rowLayout;

    public CartStoreAdapter(List storeDataList, Context mContext, int rowLayout) {
        this.storeDataList = storeDataList;
        this.mContext = mContext;
        this.rowLayout = rowLayout;
    }

    @Override
    public void onBindViewHolder(@NonNull ItemRowHolder holder, int position) {
        CartStoreModel store = storeDataList.get(position);
        holder.bind(store);
    }

    public class ItemRowHolder extends RecyclerView.ViewHolder {
        TextView tvStoreName, tvStoreTotal;
        RecyclerView rvStoreItems;

        public ItemRowHolder(@NonNull View itemView) {
            super(itemView);
            tvStoreName = itemView.findViewById(R.id.tvStoreName);
            tvStoreTotal = itemView.findViewById(R.id.tvStoreTotal);
            rvStoreItems = itemView.findViewById(R.id.rvStoreItems);
        }

        public void bind(CartStoreModel store) {
            tvStoreName.setText(store.getNama_merchant());
            Utility.currencyTXT(tvStoreTotal, String.valueOf(store.getTotalStorePrice()), mContext);

            // 子 Adapter 不再负责总价计算,仅渲染
            CartItemAdapter cartItemAdapter = new CartItemAdapter(
                store.getCartItem(), mContext, R.layout.cart_item
            );
            rvStoreItems.setAdapter(cartItemAdapter);
            rvStoreItems.setLayoutManager(new LinearLayoutManager(mContext, LinearLayoutManager.VERTICAL, false));
        }
    }
}
✅ 优势总结:数据可信:总价在进入 UI 层前已确定,不受 ViewHolder 复用、异步绑定影响;性能可控:计算可放子线程,避免卡顿;职责清晰:Adapter 专注渲染,Model/ViewModel 承担业务逻辑;易于扩展:后续支持价格实时更新、优惠券叠加等,只需修改 calculateTotalStorePrice()。

⚠️ 补充建议

  • 若需支持动态修改(如用户调整商品数量后实时刷新总价),请结合 DiffUtil + notifyItemChanged(),并在 CartItemModel 中暴露 updateQty(int) 方法,触发对应 CartStoreModel 的 calculateTotalStorePrice() 重算;
  • 避免在 Adapter 中持有 Context 强引用以防内存泄漏,建议使用 Application Context 或 WeakReference;
  • 对于超大数据集(如百级店铺),可考虑使用 ListAdapter + AsyncListDiffer 进一步优化。

通过将“计算”从 UI 层上提到数据层,你不仅解决了当前的总价传递难题,更构建了一个健壮、可维护、符合现代 Android 架构规范的购物车模块。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

564

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

547

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

153

2025.08.29

C++中int的含义
C++中int的含义

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

204

2025.08.29

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

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

588

2023.08.10

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

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

296

2023.08.14

android是什么系统
android是什么系统

Android是一种功能强大、灵活可定制、应用丰富、多任务处理能力强、兼容性好、网络连接能力强的操作系统。本专题为大家提供android相关的文章、下载、课程内容,供大家免费下载体验。

1757

2023.08.22

android权限限制怎么解开
android权限限制怎么解开

android权限限制可以使用Root权限、第三方权限管理应用程序、ADB命令和Xposed框架解开。详细介绍:1、Root权限,通过获取Root权限,用户可以解锁所有权限,并对系统进行自定义和修改;2、第三方权限管理应用程序,用户可以轻松地控制和管理应用程序的权限;3、ADB命令,用户可以在设备上执行各种操作,包括解锁权限;4、Xposed框架,用户可以在不修改系统文件的情况下修改应用程序的行为和权限。

2054

2023.09.19

抖音网页版入口与视频观看指南 抖音官网视频在线访问
抖音网页版入口与视频观看指南 抖音官网视频在线访问

本专题汇总了抖音网页版的入口链接、官方登录页面以及视频观看入口,帮助用户快速访问抖音网页版,提供免登录访问方式和直接进入视频播放页面的方法,确保顺利浏览和观看抖音视频。

33

2026.02.04

热门下载

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

精品课程

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

共23课时 | 3.2万人学习

C# 教程
C# 教程

共94课时 | 8.5万人学习

Java 教程
Java 教程

共578课时 | 57.3万人学习

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

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