当前位置: 首页 > article >正文

【商城实战(13)】购物车价格与数量的奥秘

【商城实战】专栏重磅来袭!这是一份专为开发者与电商从业者打造的超详细指南。从项目基础搭建,运用 uniapp、Element Plus、SpringBoot 搭建商城框架,到用户、商品、订单等核心模块开发,再到性能优化、安全加固、多端适配,乃至运营推广策略,102 章内容层层递进。无论是想深入钻研技术细节,还是探寻商城运营之道,本专栏都能提供从 0 到 1 的系统讲解,助力你打造独具竞争力的电商平台,开启电商实战之旅。

目录

  • 一、前端魔法:实时数量与总价的灵动之舞
    • 1.1 商品数量实时增减
    • 1.2 总价动态计算
  • 二、后端坚盾:数据一致性的稳固防线
    • 2.1 优化后端接口
    • 2.2 防止并发修改问题
  • 三、价格迷宫:促销与满减的巧妙解法
    • 3.1 促销价计算
    • 3.2 满减计算


一、前端魔法:实时数量与总价的灵动之舞

在前端的舞台上,JavaScript 是实现购物车交互的关键角色。它赋予了购物车商品数量实时增减及总价动态计算的能力,为用户带来流畅的购物体验。

1.1 商品数量实时增减

首先,我们通过监听按钮的点击事件来实现商品数量的实时增减。在 HTML 中,为每个商品的增减按钮添加唯一的标识,例如:

<button id="decrease_1">-</button>
<input type="number" id="quantity_1" value="1">
<button id="increase_1">+</button>

然后,在 JavaScript 中,使用 addEventListener 来监听按钮点击:

// 获取所有的减少按钮
const decreaseButtons = document.querySelectorAll('[id^="decrease_"]');
// 获取所有的增加按钮
const increaseButtons = document.querySelectorAll('[id^="increase_"]');
// 获取所有的数量输入框
const quantityInputs = document.querySelectorAll('[id^="quantity_"]');// 为减少按钮添加点击事件监听器
decreaseButtons.forEach((button, index) => {button.addEventListener('click', () => {let quantity = parseInt(quantityInputs[index].value);if (quantity > 1) {quantity--;quantityInputs[index].value = quantity;// 调用计算总价的函数calculateTotalPrice();}});
});// 为增加按钮添加点击事件监听器
increaseButtons.forEach((button, index) => {button.addEventListener('click', () => {let quantity = parseInt(quantityInputs[index].value);quantity++;quantityInputs[index].value = quantity;// 调用计算总价的函数calculateTotalPrice();});
});

在这段代码中,我们首先通过 querySelectorAll 方法获取所有的增减按钮和数量输入框。然后,使用 forEach 方法为每个按钮添加点击事件监听器。当点击减少按钮时,判断当前数量是否大于 1,如果是则将数量减 1 并更新输入框的值,同时调用计算总价的函数;当点击增加按钮时,直接将数量加 1 并更新输入框的值,同样调用计算总价的函数。

1.2 总价动态计算

接下来,实现总价的动态计算。我们需要获取每个商品的单价和数量,然后进行计算。假设商品单价存储在 HTML 的 data - price 属性中:

<div data - price="100" id="product_1"><!-- 商品其他信息 -->
</div>

在 JavaScript 中,计算总价的函数如下:

function calculateTotalPrice() {let totalPrice = 0;quantityInputs.forEach((input, index) => {const product = input.closest('[data - price]');const price = parseFloat(product.dataset.price);const quantity = parseInt(input.value);totalPrice += price * quantity;});// 将总价显示在页面上,假设页面上有一个id为totalPrice的元素用于显示总价document.getElementById('totalPrice').textContent = totalPrice;
}

在 calculateTotalPrice 函数中,首先初始化总价为 0。然后通过 forEach 遍历所有的数量输入框,使用 closest 方法找到对应的商品元素,获取其单价并转换为浮点数,再获取数量并转换为整数,计算出每个商品的小计并累加到总价中。最后,将计算得到的总价显示在页面上指定的元素中。通过以上前端代码的实现,我们成功地实现了购物车商品数量的实时增减及总价的动态计算,为用户提供了直观、便捷的购物体验。

二、后端坚盾:数据一致性的稳固防线

前端交互的流畅性离不开后端数据的坚实支撑。在购物车功能中,后端接口的优化对于确保商品价格、数量等数据一致性至关重要,尤其是在高并发场景下,防止并发修改问题成为关键。

2.1 优化后端接口

在处理购物车相关操作时,后端接口需要确保数据的准确性和一致性。以修改商品数量为例,传统的接口实现可能只是简单地更新数据库中的数量字段,但在并发情况下,这种方式容易出现数据不一致的问题。例如,当多个用户同时增加或减少同一商品的数量时,可能会导致最终的数量与预期不符。

为了解决这个问题,我们可以采用加锁机制。在更新商品数量前,先获取该商品对应的锁,确保同一时间只有一个线程能够进行修改操作。在 Java 中,使用 ReentrantLock 实现加锁逻辑,代码示例如下:

import java.util.concurrent.locks.ReentrantLock;public class CartService {private final ReentrantLock lock = new ReentrantLock();public void updateProductQuantity(String productId, int quantity) {lock.lock();try {// 从数据库中获取商品当前数量int currentQuantity = getCurrentQuantityFromDatabase(productId);// 更新商品数量int newQuantity = currentQuantity + quantity;// 将新数量更新到数据库updateQuantityToDatabase(productId, newQuantity);} finally {lock.unlock();}}private int getCurrentQuantityFromDatabase(String productId) {// 实际实现中,这里应包含与数据库交互的代码,查询商品当前数量并返回return 0;}private void updateQuantityToDatabase(String productId, int quantity) {// 实际实现中,这里应包含与数据库交互的代码,将商品数量更新为传入的数量}
}

在上述代码中,updateProductQuantity 方法用于更新商品数量。在方法开始时,通过 lock.lock() 获取锁,确保同一时间只有一个线程能够进入该方法。在更新商品数量的操作完成后,通过 lock.unlock() 释放锁,允许其他线程获取锁并进行操作。

2.2 防止并发修改问题

除了使用锁机制,还可以利用数据库事务来确保数据的一致性。数据库事务是一个不可分割的操作序列,要么全部执行成功,要么全部回滚。在购物车场景中,涉及商品数量、价格等数据的修改时,将这些操作放在一个事务中执行。以 MySQL 数据库为例,使用 Spring 框架的事务管理功能,配置如下:

<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><property name="dataSource" ref="dataSource"/>
</bean><tx:annotation-driven transaction-manager="transactionManager"/>

在服务层方法上添加 @Transactional 注解,将方法纳入事务管理:

import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;@Service
public class CartServiceImpl implements CartService {@Override@Transactionalpublic void updateCartItem(CartItem cartItem) {// 更新购物车中商品的数量和价格等操作// 这些操作会被视为一个事务,要么全部成功,要么全部回滚}
}

在上述配置中,DataSourceTransactionManager 配置了事务管理器,@Transactional 注解标记了需要进行事务管理的方法。当方法执行过程中出现异常时,事务会自动回滚,确保数据的一致性。通过优化后端接口和采用合适的技术手段防止并发修改问题,我们能够为购物车功能提供稳定、可靠的数据支持,保障用户的购物体验。

三、价格迷宫:促销与满减的巧妙解法

在电商的世界里,促销活动和满减策略是吸引用户的重要手段。如何在购物车中准确处理这些复杂的价格计算场景,成为了开发者需要攻克的难题。

3.1 促销价计算

当商品参与促销活动时,其价格需要按照促销规则进行计算。假设促销规则为打 8 折,我们可以在前端获取商品原价和促销折扣,然后计算促销价。在 JavaScript 中,代码实现如下:

<div data - price="100" data - discount="0.8" id="product_1"><!-- 商品其他信息 -->
</div>
function calculatePromotionPrice() {quantityInputs.forEach((input, index) => {const product = input.closest('[data - price][data - discount]');const price = parseFloat(product.dataset.price);const discount = parseFloat(product.dataset.discount);const quantity = parseInt(input.value);const promotionPrice = price * discount;// 将促销价显示在页面上,假设页面上有一个class为promotionPrice的元素用于显示促销价const promotionPriceElement = product.querySelector('.promotionPrice');if (promotionPriceElement) {promotionPriceElement.textContent = promotionPrice;}// 计算促销后的总价const totalPromotionPrice = promotionPrice * quantity;// 将促销后的总价显示在页面上,假设页面上有一个id为totalPromotionPrice的元素用于显示促销后的总价document.getElementById('totalPromotionPrice').textContent = totalPromotionPrice;});
}

在上述代码中,calculatePromotionPrice 函数用于计算商品的促销价。通过遍历所有的数量输入框,找到对应的商品元素,获取原价和折扣,计算出促销价并显示在页面上。同时,计算促销后的总价并显示。

3.2 满减计算

满减规则是电商中常见的促销方式,例如满 200 元减 50 元。在购物车总价计算中应用满减逻辑,需要先计算出购物车的总价,然后判断是否满足满减条件。在前端,使用 JavaScript 实现满减计算,代码如下:

function calculateTotalPriceWithFullReduction() {let totalPrice = 0;quantityInputs.forEach((input, index) => {const product = input.closest('[data - price]');const price = parseFloat(product.dataset.price);const quantity = parseInt(input.value);totalPrice += price * quantity;});// 满减规则:满200减50const fullReductionThreshold = 200;const fullReductionAmount = 50;let finalPrice = totalPrice;if (totalPrice >= fullReductionThreshold) {// 计算满减次数const reductionTimes = Math.floor(totalPrice / fullReductionThreshold);finalPrice = totalPrice - reductionTimes * fullReductionAmount;}// 将最终价格显示在页面上,假设页面上有一个id为finalTotalPrice的元素用于显示最终价格document.getElementById('finalTotalPrice').textContent = finalPrice;
}

在 calculateTotalPriceWithFullReduction 函数中,首先计算购物车的总价。然后定义满减规则的阈值和减免金额,判断总价是否满足满减条件。如果满足,计算满减次数并扣除相应的金额得到最终价格,最后将最终价格显示在页面上。通过以上代码的实现,我们成功地处理了购物车中商品促销价和满减等复杂的价格计算场景,为用户提供了准确的价格信息,提升了购物体验。

相关文章:

【商城实战(13)】购物车价格与数量的奥秘

【商城实战】专栏重磅来袭&#xff01;这是一份专为开发者与电商从业者打造的超详细指南。从项目基础搭建&#xff0c;运用 uniapp、Element Plus、SpringBoot 搭建商城框架&#xff0c;到用户、商品、订单等核心模块开发&#xff0c;再到性能优化、安全加固、多端适配&#xf…...

STM32之硬件SPI

SPI1和SPI2挂载的总线不一样&#xff0c;SPI1的时钟频率的比SPI2的大一倍。 核心部分是移位寄存器&#xff0c;数据一位一位的移到MOSI,同理&#xff0c;移位寄存器也一位一位的从MISO接收数据&#xff0c;LSBFIRST控制位控制高位先行还是低位先行。移位寄存器左边交叉箭头是ST…...

【Go每日一练】构建一个简单的用户信息管理系统

&#x1f47b;创作者&#xff1a;丶重明 &#x1f47b;创作时间&#xff1a;2025年3月7日 &#x1f47b;擅长领域&#xff1a;运维 目录 1.&#x1f636;‍&#x1f32b;️题目&#xff1a;简单的用户信息管理系统2.&#x1f636;‍&#x1f32b;️代码开发3.&#x1f636;‍&a…...

【力扣】2629. 复合函数——函数组合

【力扣】2629. 复合函数——函数组合 文章目录 【力扣】2629. 复合函数——函数组合题目解决方案概述方法 1&#xff1a;使用迭代的函数组合概述算法实现复杂度分析 方法 2&#xff1a;使用 Array.reduceRight() 的函数组合概述算法实现复杂度分析 附加考虑处理 this 上下文使用…...

【网络协议安全】任务10:三层交换机配置

CSDN 原创主页&#xff1a;不羁https://blog.csdn.net/2303_76492156?typeblog三层交换机是指在OSI&#xff08;开放系统互连&#xff09;模型中的第三层网络层提供路由功能的交换机。它不仅具备二层交换机的交换功能&#xff0c;还能实现路由功能&#xff0c;提供更为灵活的网…...

Linux 服务器安全配置:密码复杂度与登录超时设置

Linux服务器安全配置指南:密码复杂度与登录超时设置 一、密码复杂度设置 通过PAM模块pam_cracklib.so实现密码强度策略,配置文件: system-auth该文件主要用于定义系统范围内的认证策略,涵盖了用户登录、su 命令切换用户、sudo 权限提升等多种认证场景。当用户尝试进行系…...

依托大数据实验室建设,培育创新人才:数据科学与大数据技术专业人才培养实践

近年来&#xff0c;得益于全球大数据产业政策扶持与数字经济蓬勃发展&#xff0c;大数据市场呈现迅猛增长态势。国家层面相继出台《“数据要素”三年行动计划&#xff08;2024—2026年&#xff09;》《数字中国建设整体布局规划》等政策&#xff0c;旨在激发产业创新活力&#…...

如何使用 CSS 实现黑色遮罩效果

最近在工作中遇见了一个需求&#xff0c;鼠标经过盒子出现黑色遮罩&#xff0c;遮罩中有相关的编辑按钮&#xff0c;点击以后&#xff0c;进行图片上传并且展示&#xff0c;由于当时没有思路&#xff0c;思考了好久&#xff0c;所以在完成开发后进行总结&#xff0c;使用的技术…...

ChatGPT课件分享(37页PPT)

资料解读&#xff1a;ChatGPT课件分享 详细资料请看本解读文章的最后内容。 近年来&#xff0c;人工智能技术的迅猛发展引发了全球范围内的广泛关注&#xff0c;尤其是以OpenAI为代表的公司在自然语言处理领域的突破性进展&#xff0c;彻底改变了人机交互的方式。本文将详细解…...

开源模型时代的 AI 开发革命:Dify 技术深度解析

开源模型时代的AI开发革命&#xff1a;Dify技术深度解析 引言&#xff1a;AI开发的开源新纪元 在生成式AI技术突飞猛进的2025年&#xff0c;开源模型正成为推动行业创新的核心力量。据统计&#xff0c;全球超过80%的AI开发者正在使用开源模型构建应用&#xff0c;这一趋势不仅…...

无人机扩频技术对比!

一、技术原理与核心差异 FHSS&#xff08;跳频扩频&#xff09; 核心原理&#xff1a;通过伪随机序列控制载波频率在多个频点上快速跳变&#xff0c;收发双方需同步跳频序列。信号在某一时刻仅占用窄带频谱&#xff0c;但整体覆盖宽频带。 技术特点&#xff1a; 抗干扰…...

C语言_数据结构总结4:不带头结点的单链表

纯C语言代码&#xff0c;不涉及C 0. 结点结构 typedef int ElemType; typedef struct LNode { ElemType data; //数据域 struct LNode* next; //指针域 }LNode, * LinkList; 1. 初始化 不带头结点的初始化&#xff0c;即只需将头指针初始化为NULL即可 void Init…...

Zama TFHE-rs v1.0 发布

1. 引言 2025年2月&#xff0c;Zama 发布了 TFHE-rs v1.0&#xff0c;这是 TFHE-rs 库的第一个稳定版本。这标志着一个重要的里程碑&#xff0c;稳定了 x86 CPU 后端的高级 API&#xff0c;同时确保了向后兼容性。——即&#xff0c;现在可以依赖 TFHE-rs API&#xff0c;而不…...

AArch64架构及其编译器

—1.关于AArch64架构 AArch64是ARMv8-A架构的64位执行状态&#xff0c;支持高性能计算和大内存地址空间。它广泛应用于现代处理器&#xff0c;如苹果的A系列芯片、高通的Snapdragon系列&#xff0c;以及服务器和嵌入式设备。 • 编译器&#xff1a;可以使用GCC、Clang等编译器编…...

【ISP】对于ISP的关键算法补充

本篇是对于ISP的关键算法进行补充说明&#xff0c; 后面我们将开始逐渐深入讨论ISP的pipeline 1. 非局部均值&#xff08;NLM, Non-Local Means&#xff09; 原理 非局部均值&#xff08;NLM&#xff09;是一种基于 块匹配&#xff08;Patch Matching&#xff09; 的去噪算法…...

几种常见的虚拟环境工具(Virtualenv、Conda、System Interpreter、Pipenv、Poetry)的区别和特点总结

在 PyCharm 中创建虚拟环境是一个非常直接的过程&#xff0c;可以帮助你管理项目依赖&#xff0c;确保不同项目之间的依赖不会冲突。 通过 PyCharm 创建虚拟环境 打开 PyCharm 并选择或创建一个项目。 打开项目设置&#xff1a; 在 Windows/Linux 上&#xff0c;可以通过点击…...

Ubuntu安装问题汇总

参考文章&#xff1a; 【Ubuntu常用快捷键总结】 【王道Python常用软件安装指引】 1. 无法连接虚拟设备 sat0:0 【问题】&#xff1a;出现下图所示弹框。 【问题解决】&#xff1a; 点击 “否” 。 点击左上角的 “虚拟机” → “设置…” → “CD/DVD (SATA)” &#xff0c;…...

Ceph(1):分布式存储技术简介

1 分布式存储技术简介 1.1 分布式存储系统的特性 &#xff08;1&#xff09;可扩展 分布式存储系统可以扩展到几百台甚至几千台的集群规模&#xff0c;而且随着集群规模的增长&#xff0c;系统整体性能表现为线性增长。分布式存储的水平扩展有以下几个特性&#xff1a; 节点…...

从0开始的操作系统手搓教程43——实现一个简单的shell

目录 添加 read 系统调用&#xff0c;获取键盘输入 :sys_read putchar和clear 上班&#xff1a;实现一个简单的shell 测试上电 我们下面来实现一个简单的shell 添加 read 系统调用&#xff0c;获取键盘输入 :sys_read /* Read count bytes from the file pointed to by fi…...

【Spring】基础/体系结构/核心模块

概述&#xff1a; Spring 是另一个主流的 Java Web 开发框架&#xff0c;该框架是一个轻量级的应用框架。 Spring 是分层的 Java SE/EE full-stack 轻量级开源框架&#xff0c;以 IoC&#xff08;Inverse of Control&#xff0c;控制反转&#xff09;和 AOP&#xff08;Aspect…...

01 音视频知识学习(视频)

图像基础概念 ◼像素&#xff1a;像素是一个图片的基本单位&#xff0c;pix是英语单词picture的简写&#xff0c;加上英 语单词“元素element”&#xff0c;就得到了“pixel”&#xff0c;简称px&#xff0c;所以“像素”有“图像元素” 之意。 ◼ 分辨率&#xff1a;是指图像…...

vue3自定义hooks遇到的问题

问题 写了一个输入查询参数和url返回加载中状态、请求方法、接口返回列表的hooks&#xff0c;出现的结果是只有请求方法有效&#xff0c;加载状态无效&#xff0c;接口返回了数据&#xff0c;页面却不显示数据。 代码如下 只展示部分关键代码 import { ref, toRefs, toRef, o…...

用Python和Docker-py打造高效容器化应用管理利器

《Python OpenCV从菜鸟到高手》带你进入图像处理与计算机视觉的大门! 解锁Python编程的无限可能:《奇妙的Python》带你漫游代码世界 随着容器化技术的发展,Docker已成为现代化应用部署的核心工具。然而,手动管理容器在规模化场景下效率低下。本文深入探讨如何利用Python结…...

liunx磁盘挂载和jar启动命令

一、磁盘挂载 查看历史磁盘挂载命令&#xff1a;history | grep mount 查看所有挂载硬盘命令&#xff1a;mount 磁盘挂载命令&#xff1a;mount -t cifs -o usernamesh**,passwordP!ss**** //192.168.1.2/attachmentfilesShare2.2/pdfCert /home/nybzg/cnfai1/pdfCert 二、j…...

gbase8s rss集群通信流程

什么是rss RSS是一种将数据从主服务器复制到备服务器的方法 实例级别的复制 (所有启用日志记录功能的数据库) 基于逻辑日志的复制技术&#xff0c;需要传输大量的逻辑日志,数据库需启用日志模式 通过网络持续将数据复制到备节点 如果主服务器发生故障&#xff0c;那么备用服务…...

使用 OpenSSL 和 Python 实现 AES-256-CBC 加密与解密(安全密钥管理)

环境 OpenSSLPython 使用 OpenSSL 加密 1. 生成 AES 密钥和 IV 强烈推荐使用方法一&#xff08;Python secrets 模块&#xff09;&#xff0c;因为它更安全。 方法一: Python 的 secrets 模块&#xff08;安全方式&#xff09; 不要使用 OpenSSL 的 rand 命令直接生成密钥…...

1-001:MySQL的存储引擎有哪些?它们之间有什么区别?

MySQL 存储引擎 ├── InnoDB&#xff08;默认引擎&#xff09; │ ├── 事务支持&#xff1a;支持 ACID 和事务&#xff08;事务日志、回滚、崩溃恢复&#xff09; │ ├── 锁机制&#xff1a;支持行级锁&#xff0c;提高并发性能 │ ├── 外键支持&#xff1a;支持外键…...

持续集成与部署(CI/CD)实践指南:测试工程师的效率革命之路

一、引言 在当今快速发展的软件开发领域&#xff0c;效率和质量是至关重要的。随着软件项目的规模和复杂度不断增加&#xff0c;传统的开发和测试流程逐渐暴露出诸多问题&#xff0c;如开发周期长、集成困难、测试覆盖不足以及部署风险高等。持续集成&#xff08;Continuous I…...

C盘清理技巧分享:释放空间,提升电脑性能

目录 1. 引言 2. C盘空间不足的影响 3. C盘清理的必要性 4. C盘清理的具体技巧 4.1 删除临时文件 4.2 清理系统还原点 4.3 卸载不必要的程序 4.4 清理下载文件夹 4.5 移动大文件到其他盘 4.6 清理系统缓存 4.7 使用磁盘清理工具 4.8 清理Windows更新文件 4.9 禁用…...

如何调用 DeepSeek 的自然语言处理 API 接口并集成到在线客服系统

我在业余时间开发了一款自己的独立产品&#xff1a;升讯威在线客服与营销系统。陆陆续续开发了几年&#xff0c;从一开始的偶有用户尝试&#xff0c;到如今线上环境和私有化部署均有了越来越多的稳定用户。 随时近来 AI 大模型的火热&#xff0c;越来越多的客户&#xff0c;问…...