商城积分系统的代码实现(下)-- 积分订单的退款与结算

一、接着上文

用户在消耗积分的时候,需要根据一定的逻辑,除了扣减账户的当前余额,还需要依次消费积分订单的余额。

private void updatePointsOrderByUse(Integer schoolId, Long userId, String pointsType, int usingPoints) {List<PointsOrder> pointsOrders = pointsOrderService.listAvailablePointsOrder(schoolId, userId, pointsType);if (CollectionUtils.isNotEmpty(pointsOrders)) {for (int i = 0; i < pointsOrders.size() && usingPoints > 0; i++) {PointsOrder pointsOrder = pointsOrders.get(i);int thisUsedPoints = pointsOrder.getAvailablePoints() >= usingPoints ? usingPoints : pointsOrder.getAvailablePoints();boolean updateSuccess = this.optimisticUpdateOrder(USE_POINTS_ORDER, pointsOrder.getId(),pointsOrder.getUsedPoints(), pointsOrder.getAvailablePoints(), 0,0, pointsOrder.getAvailableSettlePoints(), thisUsedPoints,pointsOrder.getVersion());if (!updateSuccess) {if (log.isWarnEnabled()) {log.warn("积分订单处理使用事件出错, [orderNo={}, points={}]", pointsOrder.getOrderNo(), usingPoints);}Precondition.isTrue(false, "积分订单[%s]处理使用事件处理出错", pointsOrder.getOrderNo());}usingPoints -= thisUsedPoints;}}}

查询其所有的积分订单(可用余额大于0),按创建时间升序排列,也就是说,优先扣减最早的积分订单,直至全部扣减。

方法optimisticUpdateOrder()是一个乐观锁更新积分订单,和上文的方法optimisticUpdateAccount()实现类似,这里就不重复赘述了。

二、积分订单的更新逻辑

它有三个更新途径:

    // 消耗/使用积分private static final int USE_POINTS_ORDER = 1;// 积分订单的退款private static final int REFUND_POINTS_ORDER = 2;// 积分订单的结算private static final int SETTLE_POINTS_ORDER = 3;

在这里插入图片描述

1、消耗积分

更新订单的可用积分数、已使用积分数、可结算积分数

    @Modifying@Query(value = "update PointsOrder set availablePoints = :availablePoints, usedPoints = :usedPoints, " +" availableSettlePoints = :availableSettlePoints, version = version + 1, modifiedDate = now() " +" where id = :id and version = :oldVersion ")int modifyPointsOrderByUse(@Param("id") long id,@Param("availablePoints") int availablePoints,@Param("usedPoints") int usedPoints,@Param("availableSettlePoints") int availableSettlePoints,@Param("oldVersion") long oldVersion);

2、积分订单的退款

更新积分订单的已退款积分数、可用积分数

@Modifying@Query(value = "update PointsOrder set refundedPoints = :refundedPoints, " +" availablePoints = :availablePoints, version = version + 1, modifiedDate = now() " +" where id = :id and version = :oldVersion ")int modifyPointsOrderByRefund(@Param("id") long id,@Param("refundedPoints") int refundedPoints,@Param("availablePoints") int availablePoints,@Param("oldVersion") long oldVersion);

3、积分订单的结算

更新积分订单的可结算积分数、已结算积分数

    @Modifying@Query(value = "update PointsOrder set settledPoints = :settledPoints, " +" availableSettlePoints = :availableSettlePoints, version = version + 1, modifiedDate = now() " +" where id = :id and version = :oldVersion ")int modifyPointsOrderBySettle(@Param("id") long id,@Param("settledPoints") int settledPoints,@Param("availableSettlePoints") int availableSettlePoints,@Param("oldVersion") long oldVersion);

三、积分订单的退款

积分订单的结算操作,不涉及积分账户和账户收支。用户消耗积分,更新积分订单,本文的开头就已详细交待(它是和上一篇紧密相关的)

积分订单的退款则不一样,它会涉及到积分账户和账户的收支。

为了减少复杂度,我们规定积分订单只能退款一次。

既然是只能退款一次,默认就是全额退款,传入积分订单的订单号,调用本接口。

@Lock(name = POINTS_DISTRIBUTE_LOCK_PRE, key = "'refund:orderNo:' + #orderNo")@Transactional(rollbackFor = Throwable.class, isolation = Isolation.READ_COMMITTED)public void dealRefund(String orderNo) {PointsOrder pointsOrder = pointsOrderService.findPointsOrder(orderNo);Precondition.notNull(pointsOrder, "订单[%s]不存在", orderNo);// 本次退款的积分int thisPoints = pointsOrder.getAvailablePoints();if (thisPoints == 0) {if (log.isInfoEnabled()) {log.info("订单退款处理出现警告,可用积分数为0.[orderNo={}]", orderNo);}return;}// 1.更新账户的余额this.updateAccount(REFUND_POINTS_ACCOUNT, pointsOrder.getSchoolId(),pointsOrder.getUserId(), pointsOrder.getPointsType(), thisPoints);// 2.更新积分订单表boolean updateOrderSuccess = this.optimisticUpdateOrder(REFUND_POINTS_ORDER, pointsOrder.getId(),0, pointsOrder.getAvailablePoints(), pointsOrder.getRefundedPoints(),0, 0, thisPoints,pointsOrder.getVersion());if (!updateOrderSuccess) {if (log.isWarnEnabled()) {log.warn("订单退款出现错误, [orderNo={}, points={}]", orderNo, thisPoints);}Precondition.isTrue(false, "订单[%s]退款出现错误", orderNo);}//3.保存账户变更记录pointsAccountFlowService.savePointsAccountFlow(FlowTypeEnum.DECREASE,pointsOrder.getSchoolId(),pointsOrder.getUserId(),pointsOrder.getPointsType(), thisPoints,PointsChannelEnum.REFUND_ORDER.getCode(), PointsChannelEnum.REFUND_ORDER.getName(),orderNo, null,"订单号[" + orderNo + "]退款");//4.发布异步事件,通知用户其账户有变更}

四、总结

至此,关于商城的积分系统,其详细实现就介绍完了。

希望通过整个系列的五篇文章,帮助你搭建一套灵活多变的积分系统,服务于整个公司的所有业务。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/863955.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Web攻防基础篇-文件上传漏洞

文件解析安全问题上&#xff0c;格式解析是一对一的&#xff08;不能jpg解析php&#xff09;&#xff0c;换句话来说有解析错误配置或后缀解析漏洞时才能实现格式差异解析。 文件上传漏洞 程序或系统未对上传文件作全面的限制&#xff0c;导致用户可以上传某些非法文件&#…

换热器材质的选择

一、换热器材质的选择 选择烟气换热器的材质是一个涉及多个因素的综合考量过程。 1、根据烟气成分 烟气成分是一个重要的考虑因素。烟气中可能含有酸性物质和腐蚀性物质&#xff0c;如HCl、SO2等。这些物质对换热器的材质具有腐蚀性&#xff0c;因此所选材料应能够长期承受这些…

【数据结构】数据结构前置知识

这里写目录标题 基本概念与术语数据数据元素数据项数据对象数据结构 逻辑结构和物理结构物理结构顺序存储结构链式存储结构 逻辑结构集合结构线性结构树形结构图形结构 算法时间复杂度和空间复杂度大O的渐进表示法时间复杂度常数阶线性阶对数阶平方阶常见时间复杂度 空间复杂度…

Jetson系列机载电脑创建热点模式配置方法

Jetson nano为例—— 创建热点模式配置方法 1.1、新建一个 WiFi 在屏幕右上角找到网络图标&#xff0c;点击后选择“Edit Connections”选项&#xff0c;进入选择网络连接页面&#xff0c;然后点击左下角加号&#xff0c;新建一个连接&#xff0c;类型选择 WiFi 后点击 “cre…

在TkinterGUI界面显示WIFI网络(ESP32s3)摄像头画面

本实验结合了之前写过的两篇文章Python调用摄像头&#xff0c;实时显示视频在Tkinter界面以及ESP32 S3搭载OV2640摄像头释放热点&#xff08;AP&#xff09;工作模式–Arduino程序&#xff0c;当然如果手头有其他可以获得网络摄像头的URL即用于访问摄像头视频流的网络地址&…

【笔记】从零开始做一个精灵龙女-拆uv阶段

目录 先回顾一下拆uv的基础流程吧 肩部盔甲分UV示例 手环UV部分 腰带UV部分 其它也差不多&#xff0c;需要删掉一半的就先提前删掉一半&#xff0c;然后把不需要的被遮挡的面也删掉 龙角UV 胸甲UV 侧边碎发UV 马尾UV 脸部/耳朵UV 特殊情况&#xff1a;如果要删一半再…

MacOS 安装 Maven 并配置环境变量

一、简介 Maven 是一款基于 Java 平台的项目管理和整合工具&#xff0c;用来构建项目的。也就是清理、编译、测试、运行、打包、安装整个过程都交给 Maven 管理&#xff0c;整个过程就是构建。 二、安装 Java JDK Maven 依赖 Java JDK&#xff0c;如果本机没有安装过 Java 的…

Web后端开发之前后端交互

http协议 http ● 超文本传输协议 &#xff08;HyperText Transfer Protocol&#xff09;服务器传输超文本到本地浏览器的传送协议 是互联网上应用最为流行的一种网络协议,用于定义客户端浏览器和服务器之间交换数据的过程。 HTTP是一个基于TCP/IP通信协议来传递数据. HTT…

智慧校园-办公管理系统总体概述

智慧校园行政办公系统是专为高校及教育机构定制的数字化办公解决方案&#xff0c;它整合了众多办公应用与服务&#xff0c;旨在全面提升校园行政管理的效率与便捷性&#xff0c;推动信息的自由流动&#xff0c;实现绿色无纸化办公环境。该系统作为一个综合平台&#xff0c;将日…

大数据面试题之Spark(5)

Spark SQL与DataFrame的使用? Sparksql自定义函数?怎么创建DataFrame? HashPartitioner和RangePartitioner的实现 Spark的水塘抽样 DAGScheduler、TaskScheduler、SchedulerBackend实现原理 介绍下Sparkclient提交application后&#xff0c;接下来的流程? Spark的几种…

VMware中的三种虚拟网络模式

虚拟机网络模式 1 主机网络环境2 VMware中的三种虚拟网络模式2.1 桥接模式2.2 NAT模式2.3 仅主机模式 3 网络模式选择及配置NAT模式3.1 VMware虚拟网络配置3.2 虚拟机选择网络模式3.3 Windows主机网络配置 4 配置静态IP 虚拟机联网方式为桥接模式&#xff0c;这种模式下&#x…

利用OGG搭建灾备环境保姆级操作步骤

系统灾备环境搭建,使用OGG同步数据,包括&#xff1a;nfs共享磁盘、acfs新增卷组、ogg目录创建、ogg安装、ogg搭建、数据导出、导入等。 一、工作内容 1.使用NFS共享一块存储&#xff0c;用于生产环境与灾备环境之间&#xff0c;数据泵导出的dump文件存放; #生产环境新建OGG软…

c++ 设计模式 的课本范例(下)

&#xff08;19&#xff09; 桥接模式 Bridge&#xff0c;不是采用类继承&#xff0c;而是采用类组合&#xff0c;一个类的数据成员是类对象&#xff0c;来扩展类的功能。源码如下&#xff1a; class OS // 操作系统负责绘图 { public:virtual ~OS() {}virtual void draw(cha…

拼多多滑块逆向

声明(lianxi a15018601872) 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 前言(lianxi …

花生壳 搭建服务器

使用花生壳来搭建服务器。下面是一个基本的步骤指南&#xff0c;帮助您完成搭建过程&#xff1a; 注册花生壳账号&#xff1a; 访问花生壳官网&#xff08;https://hsk.oray.com/&#xff09;&#xff0c;点击注册账号。根据提示填写相关信息&#xff0c;完成账号注册。 下载并…

js懒加载

懒加载&#xff08;Lazy Loading&#xff09;是一种在网页加载过程中延迟加载某些资源的技术。在JavaScript中实现懒加载的方式有很多&#xff0c;下面是其中一种简单的实现方式&#xff1a; 1. 首先&#xff0c;给需要懒加载的元素添加一个特定的class&#xff08;例如"l…

uniapp中使用threejs加载几何体

我的建议是使用这个库 https://github.com/deepkolos/three-platformize 为什么&#xff1f;我试了uniapp推荐的和threejs-miniprogram这个小程序官方库&#xff0c;都加载不出来我的obj模型。所有我推荐不要用obj模型最好&#xff0c;挺多都支持GLTF模型的&#xff0c;但是我不…

基于ssm口红商城管理的设计与实现

一、&#x1f468;‍&#x1f393;网站题目 口红商城项目可以提供更加便捷和高效的购物方式。消费者可以在家中使用电脑或手机随时随地购物&#xff0c;避免了传统购物方式中需要花费时间和精力去实体店铺购物的麻烦。此外&#xff0c;口红商城项目还提供了更多的选择和更低的…

【Android面试八股文】性能优化相关面试题:什么时候会发生内存泄漏?举几个你遇到过的例子

内存泄漏通常发生在程序中某些对象被分配了内存但在不再需要时未能正确释放,导致这部分内存无法被垃圾回收器回收,最终造成系统内存的浪费和性能问题。 以下是一些常见的内存泄漏示例: 未关闭资源: 当使用了需要手动关闭的资源(如文件、数据库连接、网络连接等),但在使…

2D 激光 SLAM-Cartographer 实战

源码 https://github.com/cartographer-project/cartographer https://github.com/cartographer-project/cartographer_ros 课 程 下 的 注 释 版 代 码 &#xff1a; https://github.com/xiangli0608/cartographer_detailed_comments_ws 备用地址&#xff1a; https://gi…