策略设计模式-下单

news/2025/3/9 19:46:55/

1、定义一个下单context类

通过这类来判断具体使用哪个实现类,可以通过一些枚举或者条件来判断

java">import com.alibaba.fastjson.JSON;
import com.tc.common.exception.BusinessException;
import com.tc.common.user.YjkUserDetails;
import com.tc.institution.constant.RedisConstant;
import com.tc.institution.dto.req.order.OrderConfirmReq;
import com.tc.institution.dto.req.order.OrderSubmitReq;
import com.tc.institution.dto.resp.order.OrderConfirmResp;
import com.tc.institution.dto.resp.order.OrderSubmitResp;
import lombok.extern.slf4j.Slf4j;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;import javax.annotation.Resource;
import java.util.List;
import java.util.concurrent.TimeUnit;/*** TODO 4.0和4.1代码整合待完成* 1集中整合其他策略子类查询到OrderContext上下文,减少互相依赖* 2抽取已有共有方法到公共类**/
@Slf4j
@Service
public class OrderContext {@Autowiredprivate List<OrderStrategy> orderStrategyList;@Resourceprivate RedissonClient redissonClient;/*** 增加策略请维护该注释!!!* req.buyDemandOrderId 不为空走BuyDemandOrderStrategy* req.bizType = CONSULTATION_FEE(6, "问诊费订单") && req.consultationOrderId 不为空走ConsultationFeeOrderStrategy** @param req  请求参数* @param user 操作用户*/public OrderConfirmResp confirm(OrderConfirmReq req, YjkUserDetails user) {log.info("Function confirm ,req={}", req);for (OrderStrategy orderStrategy : orderStrategyList) {if (orderStrategy.confirmSupport(req) && orderStrategy.confirmCheck(req)) {OrderConfirmResp confirmResp = orderStrategy.confirm(req, user);log.info("Function confirm ,resp={}", confirmResp);return confirmResp;}}throw new BusinessException("预下单方式异常");}/*** 增加策略请维护该注释!!!* req.buyDemandOrderId 不为空走BuyDemandOrderStrategy* req.bizType = CONSULTATION_FEE(6, "问诊费订单") && req.consultationOrderId 不为空走ConsultationFeeOrderStrategy* req.bizType = CONSULTATION_EXPERT(7, "专家问诊订单") && req.consultationOrderId 不为空走ConsultationFeeOrderStrategy** @param req  请求参数* @param user 操作用户*/public OrderSubmitResp submit(OrderSubmitReq req, YjkUserDetails user) {for (OrderStrategy orderStrategy : orderStrategyList) {if (orderStrategy.submitSupport(req) && orderStrategy.submitCheck(req)) {return getOrderSubmitResp(req, user, orderStrategy);}}throw new BusinessException("预下单方式异常");}private OrderSubmitResp getOrderSubmitResp(OrderSubmitReq req, YjkUserDetails user, OrderStrategy orderStrategy) {String submitLockKey = orderStrategy.getSubmitLockKey(req, user);RLock orderLock = redissonClient.getLock(RedisConstant.ORDER_SUBMIT_LOCK_KEY + submitLockKey);try {boolean orderLockFlag = orderLock.tryLock(-1, 1, TimeUnit.MINUTES);log.info("Function submit orderLockFlag:{} param:{}", orderLockFlag, JSON.toJSONString(req));if (orderLockFlag) {OrderSubmitResp orderSubmitResp = orderStrategy.submit(req, user);log.info("Function submit param:{} result:{}", JSON.toJSONString(req), JSON.toJSONString(orderSubmitResp));return orderSubmitResp;} else {log.info("Function submit get lock fail, key:{}, now exist ...", req.getBuyDemandOrderId());throw new BusinessException("订单处理中,请勿重复多次提交");}} catch (Exception e) {log.warn("Function submit getOrderLock fail:", e);throw new BusinessException(e.getMessage());} finally {if (orderLock.isLocked() && orderLock.isHeldByCurrentThread()) {orderLock.unlock();}}}
}

2、定义一个订单策略接口,并定义公用入参

java">mport com.tc.institution.dto.req.order.OrderConfirmReq;
import com.tc.institution.dto.req.order.OrderSubmitReq;
import com.tc.institution.dto.resp.order.OrderConfirmResp;
import com.tc.institution.dto.resp.order.OrderSubmitResp;/*** 下单通用策略*/
public interface OrderStrategy<C,S,U> {/*** 预下单方式检验** @param req 预下单请求参数* @return*/boolean confirmSupport(OrderConfirmReq req);/*** 下单方式检验** @param req 下单请求参数* @return*/boolean submitSupport(OrderSubmitReq req);/*** 预下单参数校验** @param req 预下单请求参数* @return*/default boolean confirmCheck(OrderConfirmReq req) {return true;}/*** 下单参数校验** @param req 下单请求参数*/default boolean submitCheck(OrderSubmitReq req) {return true;}/*** 预下单** @param req  预下单请求参数* @param user 用户数据* @return*/OrderConfirmResp confirm(C req, U user);/*** 提交订单** @param req  下单请求参数* @param user 用户数据* @return*/OrderSubmitResp submit(S req, U user);/*** 获取提交订单锁** @param req  下单请求参数* @param user 用户数据* @return*/default String getSubmitLockKey(OrderSubmitReq req, U user) {return String.valueOf(req.getBuyDemandOrderId());}
}

3、定义不同类型下单策略实现类,根据业务可以有多个实现类

java">/*** 疫苗订单逻辑**/
@Slf4j
@Service
public class ConsultOrderStrategy implements OrderStrategy<OrderConfirmReq,OrderSubmitReq,YjkUserDetails> {}


http://www.ppmy.cn/news/1577893.html

相关文章

面试过了,总结测试工程师面试题(含答案)

普通二本计算机专业毕业&#xff0c;从毕业后&#xff0c;第一份接触测试的工作是在一家通讯小公司&#xff0c;大部分接触的工作是以功能测试为主&#xff0c;一直都是几千块钱工资&#xff0c;还一度被派出差&#xff0c;以及兼职各种产品、运维、运营的活&#xff0c;感觉自…

边缘智联新基建:5G+边缘计算重塑制造业神经末梢

一、制造业的"数据饥渴症"与旧架构的崩塌 2023年全球工业物联网设备数量已突破890亿台&#xff0c;但传统集中式云计算架构正面临三大致命瓶颈&#xff1a; 延迟黑洞&#xff1a;跨区域数据传输平均耗时达200ms以上&#xff0c;导致机器人协同、实时质检等场景无法满…

【贪心算法2】

力扣122.买卖股票最佳时机Ⅱ 链接: link 思路 要求最大利润&#xff0c;可以分解成子问题求解&#xff0c;在最低价格买入&#xff0c;最高价格卖出。 假如第0天价格最低&#xff0c;第3天价格最高&#xff0c;利润prices[3] - pricnes[0], 可以将利润公式拆解成 (prices[3]…

JavaWeb学习——Servlet介绍

Servlet 简介 什么是 Servlet Servlet 是一种服务器端的 Java 技术&#xff0c;设计用于扩展 Web 服务器或应用服务器的功能。Servlet 主要运行在服务器端&#xff0c;用来处理来自客户端的请求并生成响应。它们是 Java 技术中处理 HTTP 请求和响应的核心组件之一。 servlet…

一周热点-文本生成中的扩散模型- Mercury Coder

一、背景知识 在人工智能领域&#xff0c;文本生成模型一直是研究的热点。传统的大型语言模型多采用自回归架构&#xff0c;从左到右逐个预测下一个标记。这种模型虽然在生成连贯文本方面表现出色&#xff0c;但在速度上存在一定的局限性&#xff0c;因为它需要按顺序生成每个标…

C/C++蓝桥杯算法真题打卡(Day4)

一、P11041 [蓝桥杯 2024 省 Java B] 报数游戏 - 洛谷 算法代码&#xff1a; #include<bits/stdc.h> using namespace std;// 计算第 n 个满足条件的数 long long findNthNumber(long long n) {long long low 1, high 1e18; // 二分查找范围while (low < high) {lo…

使用ASIWebPageRequest库编写Objective-C下载器程序

使用 ASIWebPageRequest 库编写 Objective-C 下载器程序是一个简单且高效的方式来处理 HTTP 请求。在 ASIHTTPRequest 和 ASIWebPageRequest 中&#xff0c;ASIWebPageRequest 是专门用于下载网页及其资源的库。 1. 安装 ASIWebPageRequest 首先&#xff0c;你需要安装 ASIHT…

linux 设置tomcat开机启动

在Linux系统中&#xff0c;要配置Tomcat开机自启动&#xff0c;可以创建一个名为 tomcat.service 的 systemd 服务文件&#xff0c;并将其放置在 /etc/systemd/system/ 目录下。以下是一个基本的服务文件示例&#xff0c;假设Tomcat安装在 /usr/local/tomcat 路径下&#xff1a…