news 2026/1/10 12:10:18

模板方法模式 (Template Method) 在支付系统中的应用:如何优雅地对接 10+ 个第三方支付通道?

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
模板方法模式 (Template Method) 在支付系统中的应用:如何优雅地对接 10+ 个第三方支付通道?

💸 前言:被if-else支配的恐惧

场景还原:
老板:“小王,今天把微信支付接进来。”
小王:“好勒!” 于是写了一个pay()方法。
老板:“明天把支付宝也接进来。”
小王:“没问题!” 于是把pay()方法改成了:

if(channel==WECHAT){// 微信验签、调用、处理...}elseif(channel==ALIPAY){// 支付宝验签、调用、处理...}

老板:“下周我们要出海,接一下 PayPal 和 Stripe。”
小王看着那坨已经 500 行的if-else代码,陷入了沉思……

这就是典型的“缺乏抽象能力”。
支付业务虽然通道不同,但核心流程是惊人一致的

  1. 参数校验(Validate)
  2. 生成签名(Sign)
  3. 发送 HTTP 请求(Call API)
  4. 处理响应(Handle Response)
  5. 统一记录日志(Log)

针对这种**“流程固定,但细节不同”**的场景,模板方法模式就是唯一的真神。


🧠 核心原理:定义骨架,下放细节

模板方法模式的核心思想是:在父类中定义一个final的主流程方法(骨架),将具体的步骤延迟到子类去实现。

类图设计:

AbstractPaymentChannel
+pay(Request req) : void
#validate(Request req) : void
#sign(Request req) : String
#callApi(Request req, String sign) : Response
#handleResponse(Response resp) : void
-log() : void
WechatChannel
#validate()
#sign()
#callApi()
#handleResponse()
AlipayChannel
#validate()
#sign()
#callApi()
#handleResponse()

🛠️ 代码实战:重构支付网关

1. 定义抽象基类 (The Skeleton)

这是整个模式的灵魂。注意doPay方法必须是final的,防止子类篡改流程。

publicabstractclassAbstractPaymentChannel{// 核心模板方法:定义了支付的标准流程publicfinalPaymentResponsedoPay(PaymentRequestrequest){// 1. 通用日志logInfo("开始处理支付请求",request);// 2. 参数校验 (抽象步骤)if(!validate(request)){thrownewBizException("参数校验失败");}// 3. 加签 (抽象步骤)Stringsignature=sign(request);// 4. 调用三方接口 (抽象步骤)StringrawResponse=callApi(request,signature);// 5. 解析响应 (抽象步骤)PaymentResponseresponse=parseResponse(rawResponse);// 6. 通用后置处理postProcess(response);returnresponse;}// --- 抽象方法,强制子类实现 ---protectedabstractbooleanvalidate(PaymentRequestrequest);protectedabstractStringsign(PaymentRequestrequest);protectedabstractStringcallApi(PaymentRequestrequest,Stringsign);protectedabstractPaymentResponseparseResponse(StringrawResponse);// --- 通用方法,子类复用 ---privatevoidlogInfo(Stringmsg,Objectdata){// 统一的日志记录逻辑System.out.println(msg+": "+data);}// --- 钩子方法 (Hook),子类可选择性覆盖 ---protectedvoidpostProcess(PaymentResponseresponse){// 默认什么都不做}}
2. 实现微信支付通道 (Concrete Class)
@ServicepublicclassWechatPaymentChannelextendsAbstractPaymentChannel{@Overrideprotectedbooleanvalidate(PaymentRequestrequest){System.out.println("✅ 微信渠道:校验 OpenID 是否必填");returnStringUtils.hasText(request.getExtra("openId"));}@OverrideprotectedStringsign(PaymentRequestrequest){System.out.println("🔑 微信渠道:使用 MD5 进行签名");returnSecureUtil.md5(request.toString());}@OverrideprotectedStringcallApi(PaymentRequestrequest,Stringsign){System.out.println("🚀 微信渠道:调用 https://api.mch.weixin.qq.com/...");return"<xml>...SUCCESS...</xml>";}@OverrideprotectedPaymentResponseparseResponse(StringrawResponse){System.out.println("📝 微信渠道:解析 XML 响应");returnnewPaymentResponse("SUCCESS","200");}}
3. 实现支付宝通道 (Concrete Class)
@ServicepublicclassAlipayPaymentChannelextendsAbstractPaymentChannel{@Overrideprotectedbooleanvalidate(PaymentRequestrequest){System.out.println("✅ 支付宝渠道:校验 BuyerId");returntrue;}@OverrideprotectedStringsign(PaymentRequestrequest){System.out.println("🔑 支付宝渠道:使用 RSA2 进行签名");returnSecureUtil.rsa2(request.toString());}// ... 其他步骤实现}
4. 配合工厂模式使用

最后,我们需要一个简单工厂 (Simple Factory)或者策略模式 (Strategy)的 Map 来分发请求。

@ComponentpublicclassPaymentChannelFactory{@AutowiredprivateMap<String,AbstractPaymentChannel>channelMap;publicAbstractPaymentChannelgetChannel(StringchannelCode){// channelMap 会自动注入所有 Bean,key 为 beanName// 例如:wechatPaymentChannel -> WechatPaymentChannelreturnchannelMap.get(channelCode+"PaymentChannel");}}

💥 进阶技巧:钩子方法 (Hook) 的妙用

有时候,某个特定的渠道需要特殊的步骤。
比如:只有银联支付需要在支付完成后,发送短信通知用户。

我们不需要修改doPay主流程,只需要利用钩子方法

在基类中:

// 默认为空实现protectedvoidpostProcess(PaymentResponseresponse){}

在银联子类中:

@OverrideprotectedvoidpostProcess(PaymentResponseresponse){if("SUCCESS".equals(response.getStatus())){smsService.send("您的银联支付已成功!");}}

这就是“开闭原则” (Open-Closed Principle) 的完美体现:对扩展开放,对修改关闭。


📝 总结

if-else模板方法模式,不仅仅是代码行数的变化,更是思维方式的跃迁。

  1. 复用性:公共逻辑(日志、异常处理、埋点)全部收敛在父类,改一处,所有通道生效。
  2. 扩展性:接新通道?新建一个类继承父类即可,老代码一行都不用动,测试风险极低。
  3. 规范性:父类通过final关键字强制定义了业务的标准流程,新人想乱写都难。

写出机器能跑的代码是门槛,写出人能维护的代码才是本事。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2025/12/31 4:10:30

SmartDNS导致OpenWRT重启卡死?深度诊断与修复方案

SmartDNS导致OpenWRT重启卡死&#xff1f;深度诊断与修复方案 【免费下载链接】smartdns A local DNS server to obtain the fastest website IP for the best Internet experience, support DoT, DoH. 一个本地DNS服务器&#xff0c;获取最快的网站IP&#xff0c;获得最佳上网…

作者头像 李华
网站建设 2025/12/31 4:10:28

LobeChat vs ChatGPT:开源替代品能否真正媲美官方体验?

LobeChat vs ChatGPT&#xff1a;开源替代品能否真正媲美官方体验&#xff1f; 在今天&#xff0c;几乎每个接触AI的人都用过ChatGPT。它的对话自然得像真人&#xff0c;回答问题条理清晰&#xff0c;写代码、做方案、润色文案一气呵成。但当你把它引入企业环境时&#xff0c;问…

作者头像 李华
网站建设 2025/12/31 6:13:56

26、设计 SNMP MIB 全解析

设计 SNMP MIB 全解析 1. SNMP 表视图的设计考量 在设计 SNMP 表视图时,我们没有包含 id 、 edge 和 input 列,这并非疏忽。MIB 设计者(或任何接口设计者)需根据接口的预期用途来决定哪些内容有意义。在我们的案例中, edge 和 input 信息过于特定于硬件,在该…

作者头像 李华
网站建设 2025/12/31 5:22:29

HoRain云--Linux安装ShowDoc指南:IT团队的文档利器

&#x1f3ac; HoRain 云小助手&#xff1a;个人主页 ⛺️生活的理想&#xff0c;就是为了理想的生活! ⛳️ 推荐 前些天发现了一个超棒的服务器购买网站&#xff0c;性价比超高&#xff0c;大内存超划算&#xff01;忍不住分享一下给大家。点击跳转到网站。 目录 ⛳️ 推荐 …

作者头像 李华
网站建设 2025/12/31 6:13:54

33、Bash 编程:内置变量、操作符与选项全解析

Bash 编程:内置变量、操作符与选项全解析 在 Bash 编程中,内置变量、测试操作符、 set 选项、 shopt 选项以及 I/O 重定向等都是非常重要的概念。下面将对这些内容进行详细介绍。 内置 shell 变量 Bash 3.0 中有一系列可用的环境变量,这些变量在不同的场景下发挥着重…

作者头像 李华
网站建设 2025/12/31 6:13:52

单片机工程师干3年就“过气”?有人年入50万,真相就3个关键点!

单片机工程师干3年就“过气”&#xff1f;有人年入50万&#xff0c;真相就3个关键点&#xff01; 都说单片机工程师是“职场短跑道”——干三五年就薪资见顶&#xff0c;技术没挑战&#xff0c;再往上走一步比登天还难&#xff1f;但另一边&#xff0c;又有前辈靠着这行年入几十…

作者头像 李华