Java设计模式:四、行为型模式-08:策略模式
文章目录
- 一、定义:策略模式
- 二、模拟场景:策略模式
- 三、违背方案:策略模式
- 3.0 引入依赖
- 3.1 工程结构
- 3.2 优惠券折扣计算类
- 3.3 单元测试
- 四、改善代码:策略模式
- 4.1 工程结构
- 4.2 策略模式结构图
- 4.3 优惠券折扣实现
- 4.3.1 定义优惠券接口
- 4.3.2 满减优惠券接口实现
- 4.3.3 直减优惠券接口实现
- 4.3.4 折扣优惠券接口实现
- 4.3.5 n元购优惠券接口实现
- 4.3.6 策略控制类
- 4.4 单元测试
- 4.4.1 直减券测试
- 4.4.2 满减券测试
- 4.4.3 折扣券测试
- 4.4.4 n元购测试
- 五、总结:策略模式
一、定义:策略模式
- 策略模式:是具有同类可替代的行为逻辑算法场景。比如:
- 不同类型的交易方式(信用卡、支付宝、微信)。
- 生成唯一
ID
策略(UUID
、DB自增
、DB+Redis
、雪花算法
、Leaf算法
)等。
二、模拟场景:策略模式
- 模拟在购买商品时使用的各种类型优惠券(满减、直减、折扣、m元)。
- 这个场景几乎也是大家的一个日常购物省钱渠道,购买商品的时候都希望找一些优惠券,让购买的商品更加实惠。而且到了大促的时候就会有更多的优惠券需要计算那些商品一起购买更加优惠。
- 这样的场景有时候用户用起来还是蛮爽的,但是最初这样功能的设定以及产品的不断迭代,对于程序员开发还是不容易的。
- 因为这里包括了很多的规则和优惠逻辑,所以我们模拟其中的一个计算优惠的方式,使用策略模式来实现。
三、违背方案:策略模式
📖 对于优惠券的设计最初可能非常简单,就是一个金额的折扣,也没有现在这么多种类型。
所以如果没有这样场景的经验,往往设计上也是非常简单的。
但随着产品功能的不断迭代,如果程序最初设计的不具备很好的扩展性,那么往后就会越来越混乱。
3.0 引入依赖
<dependencies><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version><scope>test</scope></dependency><!-- https://mvnrepository.com/artifact/com.alibaba/fastjson --><dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.62</version></dependency><!-- LOGGING begin --><dependency><groupId>org.slf4j</groupId><artifactId>slf4j-api</artifactId><version>1.7.5</version></dependency><dependency><groupId>org.slf4j</groupId><artifactId>jcl-over-slf4j</artifactId><version>1.7.5</version></dependency><dependency><groupId>ch.qos.logback</groupId><artifactId>logback-classic</artifactId><version>1.0.9</version><exclusions><exclusion><artifactId>slf4j-api</artifactId><groupId>org.slf4j</groupId></exclusion></exclusions></dependency>
</dependencies>
3.1 工程结构
design-21.0-1
|——src|——main|--java|--com.lino.design|-CouponDiscountService.java|--test|--com.lino.design.test|-ApiTest.java
3.2 优惠券折扣计算类
CouponDiscountService.java
package com.lino.design;/*** @description: 优惠券折扣计算接口*/
public class CouponDiscountService {/*** 计算优惠券折扣** @param type 优惠券类型:1-直减券,2-满减券,3-折扣券,4-n元购* @param typeContent 折扣价格* @param skuPrice 商品价格* @param typeExt 满减价格* @return 折扣后的价格*/public double discountAmount(int type, double typeContent, double skuPrice, double typeExt) {// 1.直减券if (1 == type) {return skuPrice - typeContent;}// 2.满减券if (2 == type) {if (skuPrice < typeExt) {return skuPrice;}return skuPrice - typeContent;}// 3.折扣券if (3 == type) {return skuPrice * typeContent;}// 4.n元购if (4 == type) {return typeContent;}return 0D;}
}
- 以上是不同类型的优惠券计算折扣后的实际金额。
- 入参包括:优惠券类型、优惠券金额、商品金额、满减金额
- 因为有些优惠券是满多少减少多少,所以增加了
typeExt
类型,这也是方法的不好扩展性问题。
- 因为有些优惠券是满多少减少多少,所以增加了
- 最后是整个方法体中对优惠券折扣金额的实现,最开始可能是一个最简单的优惠券,后面随着产品功能的增加,不断的扩展
if
语句。
3.3 单元测试
ApiTest.java
package com.lino.design.test;import com.lino.design.CouponDiscountService;
import org.junit.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;/*** @description: 单元测试*/
public class ApiTest {private Logger logger = LoggerFactory.getLogger(ApiTest.class);@Testpublic void test() {CouponDiscountService couponDiscountService = new CouponDiscountService();double result1 = couponDiscountService.discountAmount(1, 10D, 100D, 0D);logger.info("测试结果:直减优惠后金额:{}", result1);double result2 = couponDiscountService.discountAmount(2, 10D, 100D, 0D);logger.info("测试结果:满减优惠后金额:{}", result2);double result3 = couponDiscountService.discountAmount(3, 0.9D, 100D, 0D);logger.info("测试结果:折扣优惠后金额:{}", result3);double result4 = couponDiscountService.discountAmount(4, 90D, 100D, 0D);logger.info("测试结果:n元购金额:{}", result4);}
}
测试结果
17:05:07.040 [main] INFO com.lino.design.test.ApiTest - 测试结果:直减优惠后金额:90.0
17:05:07.049 [main] INFO com.lino.design.test.ApiTest - 测试结果:满减优惠后金额:90.0
17:05:07.049 [main] INFO com.lino.design.test.ApiTest - 测试结果:折扣优惠后金额:90.0
17:05:07.049 [main] INFO com.lino.design.test.ApiTest - 测试结果:n元购金额:90.0
四、改善代码:策略模式
💡 重构使用策略模式,优化代码结构,增强整体的扩展性。
4.1 工程结构
design-21.0-2
|——src|——main|--java|--com.lino.design|--impl| |--MJCouponDiscount.java| |--NYGCouponDiscount.java| |--ZJCouponDiscount.java| |--ZKCouponDiscount.java|-Context.java|-ICouponDiscount.java|--test|--com.lino.design.test|-ApiTest.java
4.2 策略模式结构图
- 整体的结构模式并不复杂,主要体现的不同类型的优惠券在计算优惠券方式的不同计算策略。
- 这里包括一个接口类(
ICouponDiscount
),以及四种优惠券类型的实现方式。 - 最后提供了策略模式的上下控制类处理,整体的策略服务。
4.3 优惠券折扣实现
4.3.1 定义优惠券接口
ICouponDiscount.java
package com.lino.design;import java.math.BigDecimal;/*** @description: 优惠券折扣计算接口*/
public interface ICouponDiscount<T> {/*** 计算优惠券折扣** @param couponInfo 优惠券信息泛型* @param skuPrice 商品价格* @return 优惠券折扣后的价格*/BigDecimal discountAmount(T couponInfo, BigDecimal skuPrice);
}
- 定义了优惠券折扣接口,也增加了泛型用于不同类型的接口可以传递不同的类型参数。
- 接口中包括商品金额以及出参返回最终折扣后的金额。
4.3.2 满减优惠券接口实现
MJCouponDiscount.java
package com.lino.design.impl;import com.lino.design.ICouponDiscount;
import java.math.BigDecimal;
import java.util.Map;/*** @description: 满减券*/
public class MJCouponDiscount implements ICouponDiscount<Map<String, String>> {/*** 满减计算* 1.判断满足x元后-n元,否则不减* 2.最低 支付金额1元** @param couponInfo 优惠券信息泛型* @param skuPrice 商品价格* @return 优惠后价格*/@Overridepublic BigDecimal discountAmount(Map<String, String> couponInfo, BigDecimal skuPrice) {String x = couponInfo.get("x");String n = couponInfo.get("n");// 小于商品金额条件的,直接返回商品原价if (skuPrice.compareTo(new BigDecimal(x)) < 0) {return skuPrice;}// 减去优惠金额判断BigDecimal discountAmount = skuPrice.subtract(new BigDecimal(n));if (discountAmount.compareTo(BigDecimal.ZERO) < 1) {return BigDecimal.ONE;}return discountAmount;}
}
4.3.3 直减优惠券接口实现
ZJCouponDiscount.java
package com.lino.design.impl;import com.lino.design.ICouponDiscount;
import java.math.BigDecimal;/*** @description: 直减券*/
public class ZJCouponDiscount implements ICouponDiscount<Double> {/*** 直减计算* 1.使用商品价格减去优惠价格* 2.最低支付金额1元** @param couponInfo 优惠券信息泛型* @param skuPrice 商品价格* @return 优惠后价格*/@Overridepublic BigDecimal discountAmount(Double couponInfo, BigDecimal skuPrice) {BigDecimal discountAmount = skuPrice.subtract(new BigDecimal(couponInfo));if (discountAmount.compareTo(BigDecimal.ZERO) < 1) {return BigDecimal.ONE;}return discountAmount;}
}
4.3.4 折扣优惠券接口实现
ZKCouponDiscount.java
package com.lino.design.impl;import com.lino.design.ICouponDiscount;
import java.math.BigDecimal;/*** @description: 折扣券*/
public class ZKCouponDiscount implements ICouponDiscount<Double> {/*** 折扣计算* 1.使用商品价格乘以折扣比例,为最后支付金额* 2.保留两位小数* 3.最低支付金额1元** @param couponInfo 优惠券信息泛型* @param skuPrice 商品价格* @return 优惠后价格*/@Overridepublic BigDecimal discountAmount(Double couponInfo, BigDecimal skuPrice) {BigDecimal discountAmount = skuPrice.multiply(new BigDecimal(couponInfo)).setScale(2, BigDecimal.ROUND_HALF_UP);if (discountAmount.compareTo(BigDecimal.ZERO) < 1) {return BigDecimal.ONE;}return discountAmount;}
}
4.3.5 n元购优惠券接口实现
NYGCouponDiscount.java
package com.lino.design.impl;import com.lino.design.ICouponDiscount;
import java.math.BigDecimal;
import java.util.Map;/*** @description: n元购*/
public class NYGCouponDiscount implements ICouponDiscount<Double> {/*** n元购* 1.无论原价多少钱都固定金额购买** @param couponInfo 优惠券信息泛型* @param skuPrice 商品价格* @return 优惠后价格*/@Overridepublic BigDecimal discountAmount(Double couponInfo, BigDecimal skuPrice) {return new BigDecimal(couponInfo);}
}
4.3.6 策略控制类
Context.java
package com.lino.design;import java.math.BigDecimal;/*** @description: 策略控制类*/
public class Context<T> {private ICouponDiscount<T> couponDiscount;public Context(ICouponDiscount<T> couponDiscount) {this.couponDiscount = couponDiscount;}/*** 计算优惠券折扣** @param couponInfo 优惠券信息泛型* @param skuPrice 商品价格* @return 优惠券折扣后的价格*/public BigDecimal discountAmount(T couponInfo, BigDecimal skuPrice) {return couponDiscount.discountAmount(couponInfo, skuPrice);}
}
- 策略模式的控制类主要是外部可以传递不同的策略实现,再通过统一的方法执行优惠策略计算。
- 另外这里也可以包装出
map
结构,让外部只需要对应的泛型类型即可使用相应的服务。
4.4 单元测试
4.4.1 直减券测试
ApiTest.java
@Test
public void test_zj() {Context<Double> context = new Context<>(new ZJCouponDiscount());BigDecimal discountAmount = context.discountAmount(10D, new BigDecimal(100));logger.info("测试结果:直减优惠后金额:{}", discountAmount);
}
测试结果
17:16:00.390 [main] INFO com.lino.design.test.ApiTest - 测试结果:直减优惠后金额:90
4.4.2 满减券测试
ApiTest.java
@Test
public void test_mj() {Context<Map<String, String>> context = new Context<>(new MJCouponDiscount());Map<String, String> mapReq = new HashMap<>();mapReq.put("x", "100");mapReq.put("n", "10");BigDecimal discountAmount = context.discountAmount(mapReq, new BigDecimal(100));logger.info("测试结果:满减优惠后金额:{}", discountAmount);
}
测试结果
17:16:35.300 [main] INFO com.lino.design.test.ApiTest - 测试结果:满减优惠后金额:90
4.4.3 折扣券测试
ApiTest.java
@Test
public void test_zk() {Context<Double> context = new Context<>(new ZKCouponDiscount());BigDecimal discountAmount = context.discountAmount(0.9D, new BigDecimal(100));logger.info("测试结果:折扣9折后金额:{}", discountAmount);
}
测试结果
17:17:06.907 [main] INFO com.lino.design.test.ApiTest - 测试结果:折扣9折后金额:90.00
4.4.4 n元购测试
ApiTest.java
@Test
public void test_nyg() {Context<Double> context = new Context<>(new NYGCouponDiscount());BigDecimal discountAmount = context.discountAmount(90D, new BigDecimal(100));logger.info("测试结果:n元购优惠后金额:{}", discountAmount);
}
测试结果
17:17:35.616 [main] INFO com.lino.design.test.ApiTest - 测试结果:n元购优惠后金额:90
💡 以上四个测试分别验证了不同类型优惠券的优惠策略,测试结果是满足我们的预期。
这里四种优惠券最终都是再原价100
元上折扣10
元,最终支付90
元。
五、总结:策略模式
- 策略模式案例相对来说并不复杂,主要的逻辑都是体现在关于不同类型优惠券的计算折扣策略上。
- 结构相对来说也比较简单,在实际的开发中这样的设计模式也是非常常用的。
- 另外策略模式与命令模式、适配器模式结构相似,但是思路是有差异的。
- 通过策略模式的使用可以把我们方法中的
if
语句优化掉,大量的if
语句使用会让代码难为扩展,也不好维护,同时在后期遇到各种问题也很难维护。 - 在使用策略模式可以很好的满足隔离性和扩展性,对于不断新增的需求也非常方便承接。
策略模式
、适配器模式
、组合模式
等,在一些结构上是比较相似的。但是每一个模式都有自己的逻辑特点,在使用的过程中最佳的方式是经过较多的实践来吸取经验,为后续的研发设计提供更好的技术输出。
相关文章:

Java设计模式:四、行为型模式-08:策略模式
文章目录 一、定义:策略模式二、模拟场景:策略模式三、违背方案:策略模式3.0 引入依赖3.1 工程结构3.2 优惠券折扣计算类3.3 单元测试 四、改善代码:策略模式4.1 工程结构4.2 策略模式结构图4.3 优惠券折扣实现4.3.1 定义优惠券接…...

ICCOA蓝牙数字车钥匙2.0
近期,ICCOA智慧车联开放联盟宣布,由小米、vivo、OPPO、长安、深蓝、吉利、极氪、比亚迪、北汽、蔚来、零跑、雪球、瓶钵、融卡等企业联合参与制定的ICCOA蓝牙数字车钥匙2.0标准(简称DK2.0),于2023年9月1日正式发布实施…...

ArcGIS土地利用程度综合指数分析
成图展示: 土地利用程度综合指数 第一步 准备数据 使用的数据为2010年河南省土地利用类型数据与其行政区划县级数据(为了节省操作,这里使用上次实验的部分数据[1],各土地利用类型已被提取) 第二步 面积统计 水域为例…...

服务端请求伪造(SSRF)及漏洞复现
文章目录 渗透测试漏洞原理服务端请求伪造1. SSRF 概述1.1 SSRF 场景1.1.1 PHP 实现 1.2 SSRF 原理1.3 SSRF 危害 2. SSRF 攻防2.1 SSRF 利用2.1.1 文件访问2.1.2 端口扫描2.1.3 读取本地文件2.1.4 内网应用指纹识别2.1.5 攻击内网Web应用 2.2 SSRF 经典案例2.2.1 访问页面2.2.…...
v-model和v-bind
v-model,它其实就是一个语法糖,作用就是双向绑定表单控件(radio, text,address,email,select,checkbox,textarea) v-bind(简写形式:value值),用于绑定属性值,只能实现数据的单项绑定。 <template> <div>…...

详细介绍 弹性盒子(display:flex)
文章目录 什么是弹性盒子 如何使用弹性盒子flex系列flex-direction 对齐方向 水平对齐垂直对齐flex-wrap 换行flex-flowflex模型说明容器的属性 justify-content X轴对齐方式align-content Y轴对齐方式总结属性值Y轴对齐的另外一种:align-itemsalign-content和alig…...

Docker使用数据卷挂载进行数据存储与共享
一、挂载和数据卷 在 Docker 中,挂载(Mounting)和数据卷(Data Volumes)是用于在容器和宿主机之间共享数据的机制。 挂载:将宿主机文件系统中的目录或文件与容器中的目录或文件进行关联的过程。数据卷&…...
[力扣146. LRU 缓存 ](https://leetcode.cn/problems/lru-cache/description/)
力扣146. LRU 缓存 使用LinkedHashmap(HashMap的子类,能够记住插入数据的顺序). LRU是Lease Recently User的缩写,意思是最近 最少使用。比如设计一个文件缓存系统,每个文件有自己的大小和访问时间,文件缓存系统有总的大小&…...
Mysql存储引擎
目录 一、前言 二、存储引擎 1、InnoDB存储引擎 1.1、简介 1.2、优势 1.3、使用方法 1.4、性能 2、MyISAM存储引擎 2.1、优势 2.2、使用方法 2.3、性能 3、MEMORY存储引擎 4、MyISAM 三、比较 四、总结 一、前言 MySQL是开放源代码的关系型数据库管理系统&…...
算法通关村-----数组实现加法专题问题解析
数组实现整数加法 问题描述 给定一个由 整数 组成的 非空 数组所表示的非负整数,在该数的基础上加一。最高位数字存放在数组的首位, 数组中每个元素只存储单个数字。你可以假设除了整数 0 之外,这个整数不会以零开头。详见leetcode66 问题…...

倒排表的压缩算法
For压缩算法 这是倒排表的一种压缩算法。 还是那个问题,如果"小米" 这个词项,在多文档里都有,则就会导致倒排表很大,这时候就会设计到了压缩算法,这里说的是,倒排表。 那末我们来看看 for压缩…...

Android studio实现自定义圆形进度条 带刻度进度条 计步效果 时速表 水波纹效果
目录 原文链接效果图values /layout /activity原文链接 效果图 点击重置后: 该项目总共实现了三种圆形进度条效果 CircleProgress:圆形进度条,可以实现仿 QQ 健康计步器的效果,支持配置进度条背景色、宽度、起始角度,支持进度条渐变DialProgress:类似 CircleProgress,…...

使用【宝塔+docker】在云服务器上部署基于SpringBoot 和 Dubbo RPC 的项目:踩坑记录
待部署的项目包括:前端front,服务提供者backend,服务消费者gateway,注册中心nacos 服务器信息:腾讯云入门级服务器2核2G(后续有对服务器进行升级) 部署工具:前端使用宝塔部署&#x…...

【算法与数据结构】617、LeetCode合并二叉树
文章目录 一、题目二、解法三、完整代码 所有的LeetCode题解索引,可以看这篇文章——【算法和数据结构】LeetCode题解。 一、题目 二、解法 思路分析:采用递归的方式遍历二叉树,【算法与数据结构】144、94、145LeetCode二叉树的前中后遍历&am…...

ffmpeg把RTSP流分段录制成MP4,如果能把ffmpeg.exe改成ffmpeg.dll用,那音视频开发的难度直接就降一个维度啊
比如,原来我们要用ffmpeg录一段RTSP视频流转成MP4,我们有两种方案: 方案一:可以使用以下命令将rtsp流分段存储为mp4文件 ffmpeg -i rtsp://example.com/stream -vcodec copy -acodec aac -f segment -segment_time 3600 -reset_t…...

朝夕光年游戏自动化测试实践
朝夕光年是面向全球用户与开发者的游戏研发与发行业务品牌,致力于服务全球玩家,帮助玩家在令人惊叹的虚拟世界中一起玩耍与创造。 在游戏的研发过程中,游戏自动化一直是开展难度较大的工程,具体包括机房机架、设备调度、软件框架、…...

数据结构基本概念
一、数据 数据对象-数据元素-数据项(属性),前者由后者组成 二、数据结构 定义:按某种关系的数据元素的集合 三、数据类型 1、原子类型(例如整型) 2、结构类型(由原子类型组成,例如数组) 3、…...

【javaweb】学习日记Day9 - Mybatis 基础操作
目录 一、删除 (1)在mapper接口执行sql删除语句 ① 注解后sql语句没有提示怎么办? (2)测试层 (3)开启mybatis日志 (4)预编译SQL 二、新增 (1&#…...

Mybatis学习|Mybatis缓存:一级缓存、二级缓存
Mybatis缓存 MyBatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大的提升查询效率。 MyBatis系统中默认定义了两级缓存:一级缓存和二级缓存 默认情况下,只有一级缓存开启。(SqlSession级别的缓存,也称为本地…...
230903文本docx
处理文本 块级项目,每次文本超出右边界时都会添加一行.对段落,边界一般是页边距,但如果按列布局页,则也可是列边界,如果表格单元格内有段,则也可是单元格边界. 块级项属性指定其在页上的位置,如缩进项及段落前后间距.内联项属性一般指定显示内容的如字样,字体大小,粗体和斜体…...
Leetcode 3576. Transform Array to All Equal Elements
Leetcode 3576. Transform Array to All Equal Elements 1. 解题思路2. 代码实现 题目链接:3576. Transform Array to All Equal Elements 1. 解题思路 这一题思路上就是分别考察一下是否能将其转化为全1或者全-1数组即可。 至于每一种情况是否可以达到…...

阿里云ACP云计算备考笔记 (5)——弹性伸缩
目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...

JavaScript 中的 ES|QL:利用 Apache Arrow 工具
作者:来自 Elastic Jeffrey Rengifo 学习如何将 ES|QL 与 JavaScript 的 Apache Arrow 客户端工具一起使用。 想获得 Elastic 认证吗?了解下一期 Elasticsearch Engineer 培训的时间吧! Elasticsearch 拥有众多新功能,助你为自己…...

linux arm系统烧录
1、打开瑞芯微程序 2、按住linux arm 的 recover按键 插入电源 3、当瑞芯微检测到有设备 4、松开recover按键 5、选择升级固件 6、点击固件选择本地刷机的linux arm 镜像 7、点击升级 (忘了有没有这步了 估计有) 刷机程序 和 镜像 就不提供了。要刷的时…...

学习STC51单片机31(芯片为STC89C52RCRC)OLED显示屏1
每日一言 生活的美好,总是藏在那些你咬牙坚持的日子里。 硬件:OLED 以后要用到OLED的时候找到这个文件 OLED的设备地址 SSD1306"SSD" 是品牌缩写,"1306" 是产品编号。 驱动 OLED 屏幕的 IIC 总线数据传输格式 示意图 …...

ETLCloud可能遇到的问题有哪些?常见坑位解析
数据集成平台ETLCloud,主要用于支持数据的抽取(Extract)、转换(Transform)和加载(Load)过程。提供了一个简洁直观的界面,以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...
Spring AI 入门:Java 开发者的生成式 AI 实践之路
一、Spring AI 简介 在人工智能技术快速迭代的今天,Spring AI 作为 Spring 生态系统的新生力量,正在成为 Java 开发者拥抱生成式 AI 的最佳选择。该框架通过模块化设计实现了与主流 AI 服务(如 OpenAI、Anthropic)的无缝对接&…...

微信小程序云开发平台MySQL的连接方式
注:微信小程序云开发平台指的是腾讯云开发 先给结论:微信小程序云开发平台的MySQL,无法通过获取数据库连接信息的方式进行连接,连接只能通过云开发的SDK连接,具体要参考官方文档: 为什么? 因为…...
浅谈不同二分算法的查找情况
二分算法原理比较简单,但是实际的算法模板却有很多,这一切都源于二分查找问题中的复杂情况和二分算法的边界处理,以下是博主对一些二分算法查找的情况分析。 需要说明的是,以下二分算法都是基于有序序列为升序有序的情况…...

Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决
Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决 问题背景 在一个基于 Spring Cloud Gateway WebFlux 构建的微服务项目中,新增了一个本地验证码接口 /code,使用函数式路由(RouterFunction)和 Hutool 的 Circle…...