0

0

如何在嵌套 RecyclerView 中准确传递和计算多层商品总价

聖光之護

聖光之護

发布时间:2026-02-04 18:52:02

|

229人浏览过

|

来源于php中文网

原创

如何在嵌套 RecyclerView 中准确传递和计算多层商品总价

本文详解在三层嵌套 recyclerview(店铺 → 商品 → 增值项)场景下,如何可靠地汇总每家店铺的总金额(含商品主价、数量及附加项价格),避免因 adapter 生命周期与数据绑定时机导致的 `totalitems` 丢失问题。

在 Android 开发中,使用嵌套 RecyclerView 实现多级列表(如购物车按店铺分组 → 每店多个商品 → 每商品多个可选附加项)虽灵活,但极易陷入「状态不可靠」陷阱——正如你所遇到的:CartItemAdapter 内部 totalItems 在 ViewHolder 的 bind() 中累加正常(Log 输出验证无误),但调用 getTotalItems() 却返回 0.0。根本原因在于:RecyclerView Adapter 的 onBindViewHolder() 是按需触发、可复用、非顺序执行的;totalItems 字段未在数据源层面固化,且其累加逻辑依赖于 UI 渲染过程(即 bind() 调用),而该过程不可控、不保证完成、甚至可能被回收或跳过(如 Item 不可见时)

因此,绝不能将业务关键数据(如总价)的计算和存储耦合在 Adapter 或 ViewHolder 的渲染生命周期中。正确做法是:将总价计算前移至数据准备阶段,在构建 CartStoreModel 时就完成所有层级的聚合,并将其作为只读字段持久化到模型中

✅ 推荐方案:预计算 + 不可变模型 + ViewModel 协调

1. 扩展 CartStoreModel,添加预计算总价字段

public class CartStoreModel {
    private String nama_merchant;
    private List cartItem;
    private double storeTotalPrice; // ← 新增:店铺级预计算总价(含所有商品+附加项)

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

    // Getter
    public double getStoreTotalPrice() { return storeTotalPrice; }
}

2. 在 ViewModel 中统一预处理(推荐异步)

public class CartViewModel extends ViewModel {
    private MutableLiveData> cartStoreList = new MutableLiveData<>();

    public LiveData> getCartStoreList() {
        return cartStoreList;
    }

    // 在 Repository 或 DataLayer 获取原始数据后调用
    public void loadAndCalculateCart() {
        // 示例:模拟从本地/网络获取原始数据
        List rawStores = fetchRawCartData();

        // 异步计算总价,避免阻塞主线程(尤其数据量大时)
        Executors.newSingleThreadExecutor().execute(() -> {
            for (CartStoreModel store : rawStores) {
                store.initTotalPrice(); // 关键:此处完成所有嵌套层级的累加
            }

            // 切回主线程更新 UI
            new Handler(Looper.getMainLooper()).post(() -> 
                cartStoreList.setValue(rawStores)
            );
        });
    }
}

3. 修改 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.getStoreTotalPrice()), mContext);

            // 子 Adapter 不再负责计算,仅负责展示
            CartItemAdapter itemAdapter = new CartItemAdapter(store.getCartItem(), mContext, R.layout.cart_item);
            rvStoreItems.setAdapter(itemAdapter);
            rvStoreItems.setLayoutManager(new LinearLayoutManager(mContext, LinearLayoutManager.VERTICAL, false));
        }
    }
}
⚠️ 关键注意事项 删除 CartItemAdapter 中的 totalItems 字段及累加逻辑:它已失去存在意义,且易引发线程安全与状态不一致问题。 CartItemAdapter 和 CartItemAddonAdapter 应保持纯展示职责:它们只读取 CartItemModel 和 CartItemVarModel 中已计算好的字段(如 subTotal 可提前存入模型)。 避免在 onBindViewHolder() 中做耗时计算:即使数据量小,也应遵循单一职责原则,将计算与渲染分离。 若需实时响应数量/选项变更:监听子项事件(如 btnAdd 点击),通过回调通知 ViewModel 重新计算并刷新对应 CartStoreModel 的 storeTotalPrice,再 notifyItemChanged()。

✅ 总结

嵌套 RecyclerView 的总价传递本质是数据流设计问题,而非 UI 绑定技巧问题。正确的解法是:

Viggle AI
Viggle AI

Viggle AI是一个AI驱动的3D动画生成平台,可以帮助用户创建可控角色的3D动画视频。

下载
  1. 数据先行:在数据进入 UI 层之前,由 ViewModel 或 Repository 完成全部聚合计算;
  2. 模型承载:将计算结果作为不可变(或受控可变)字段存入数据模型;
  3. Adapter 只读:Adapter 及其 ViewHolder 仅负责高效、安全地读取并渲染这些预计算值;
  4. 响应式更新:用户交互触发变更时,走“事件 → ViewModel 重算 → 更新模型 → 刷新 UI”闭环。

此方案彻底规避了 onBindViewHolder 的不确定性,提升代码可测试性、可维护性与运行稳定性,是处理复杂嵌套列表业务逻辑的工业级实践。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

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

android重启应用的方法有哪些
android重启应用的方法有哪些

android重启应用有通过Intent、PendingIntent、系统服务、Runtime等方法。本专题为大家提供Android相关的文章、下载、课程内容,供大家免费下载体验。

277

2023.10.18

Android语音播放功能实现方法
Android语音播放功能实现方法

实现方法有使用MediaPlayer实现、使用SoundPool实现两种。可以根据具体的需求选择适合的方法进行实现。想了解更多语音播放的相关内容,可以阅读本专题下面的文章。

352

2024.03.01

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

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

32

2026.02.04

学习通网页版入口与在线学习指南 学习通官网登录与使用方法
学习通网页版入口与在线学习指南 学习通官网登录与使用方法

本专题详细汇总了学习通网页版入口与登录方法,提供学习通官方网页端入口、学生登录平台、网页版使用指南等内容,帮助用户快速稳定地登录学习通官网,顺利进入学习平台,提升学习效率和体验。

6

2026.02.04

Python Web 框架 Django 深度开发
Python Web 框架 Django 深度开发

本专题系统讲解 Python Django 框架的核心功能与进阶开发技巧,包括 Django 项目结构、数据库模型与迁移、视图与模板渲染、表单与认证管理、RESTful API 开发、Django 中间件与缓存优化、部署与性能调优。通过实战案例,帮助学习者掌握 使用 Django 快速构建功能全面的 Web 应用与全栈开发能力。

7

2026.02.04

热门下载

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

精品课程

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

共162课时 | 15.4万人学习

Java 教程
Java 教程

共578课时 | 57.4万人学习

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

共64课时 | 6.8万人学习

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

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