UniApp + SpringBoot 實現支付寶支付和退款功能
上篇介紹瞭UniApp + SpringBoot 實現微信支付和退款功能,喜歡的朋友可以點擊查看。
開發準備
- 一臺用於支付的測試機
- 用於編寫的後端框架接口的 IDE (IDEA 或者 Eclipse 都可以)
- HBuilder X 用來編輯 UniApp 項目的編輯器和編譯器
- 基本的 SpringBoot 的腳手架,可以去 https://start.spring.io/ 或者 IDEA 自帶的快速生成腳手架插件。
- Jdk 11
支付寶支付開發
後端部分
在 SpringBoot 中添加以下坐標
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!-- 支付寶官方 SDK--> <dependency> <groupId>com.alipay.sdk</groupId> <artifactId>alipay-sdk-java</artifactId> <version>4.22.32.ALL</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-configuration-processor</artifactId> <optional>true</optional> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <optional>true</optional> </dependency>
在 resources 目錄下添加 application.yml 我們不去用默認的 application.properties 文件,畢竟 yml 更好看點。並在 yml 中添加以下內容
# 服務啟動端口 server: port: 8080 # 支付寶支付 alipay: server_url: https://openapi.alipay.com/gateway.do app_id: 企業支付的 APPID private_key: 申請的私鑰 format: json charset: utf-8 alipay_public_key: 申請的公鑰 sign_type: RSA2 notifyUrl: 回調地址
創建一個 AlipayConfig.java 繼承 com.alipay.api.AlipayConfig
@Getter @Setter @ToString @Component @ConfigurationProperties(prefix = "alipay") public class AlipayConfig extends com.alipay.api.AlipayConfig { private String serverUrl; private String appId; private String privateKey; private String format; private String charset; private String alipayPublicKey; private String signType; private String notifyUrl; }
創建一個 BizAlipayService.java
import com.alipay.api.AlipayApiException; import com.alipay.api.DefaultAlipayClient; import com.alipay.api.domain.AlipayTradeAppPayModel; import com.alipay.api.request.AlipayTradeAppPayRequest; import com.alipay.api.response.AlipayTradeAppPayResponse; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; /** * 阿裡雲支付類 */ @Service public class BizAlipayService { private static Logger logger = LoggerFactory.getLogger(BizAlipayService.class); @Autowired AlipayConfig alipayConfig; private DefaultAlipayClient client() throws AlipayApiException { return new DefaultAlipayClient(alipayConfig); } /** * 預下單 * * @param subject 訂單標題 * @param outTradeNo 商傢生成的訂單號 * @param totalAmount 訂單總價值 * @return */ public String appPay(String subject, String outTradeNo, String totalAmount) { String source = ""; try { DefaultAlipayClient client = client(); AlipayTradeAppPayModel model = new AlipayTradeAppPayModel(); model.setSubject(subject); model.setOutTradeNo(outTradeNo); model.setTotalAmount(totalAmount); // alipay 封裝的接口調用 AlipayTradeAppPayRequest request = new AlipayTradeAppPayRequest(); request.setBizModel(model); request.setNotifyUrl(alipayConfig.getNotifyUrl()); AlipayTradeAppPayResponse response = client.sdkExecute(request); source = response.getBody(); } catch (AlipayApiException e) { logger.error("支付出現問題,詳情:{}", e.getErrMsg()); e.printStackTrace(); } return source; } }
創建一個 AlipayController.java 來實現接口給前端調用時使用
import com.alipay.api.AlipayApiException; import com.alipay.api.internal.util.AlipaySignature; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RestController; import javax.servlet.http.HttpServletRequest; import java.math.BigDecimal; import java.util.HashMap; import java.util.Map; @RestController public class AlipayController { private static Logger logger = LoggerFactory.getLogger(AlipayController.class); @Autowired AlipayConfig alipayConfig; @Autowired BizAlipayService alipayService; /** * 支付接口 * * @return */ @GetMapping("/pay") public String orderPay() { String s = alipayService.appPay("測試支付", String.valueOf(System.currentTimeMillis()), new BigDecimal("0.01").toString()); logger.info("支付生成信息:{}", s); return s; } /** * 訂單回調 * * @return */ @RequestMapping(method = RequestMethod.POST, value = "/notify") public String orderNotify(HttpServletRequest request) { Map<String, String> params = new HashMap<>(); Map<String, String[]> requestParams = request.getParameterMap(); for (String name : requestParams.keySet()) { String[] values = requestParams.get(name); String valueStr = ""; for (int i = 0; i < values.length; i++) { valueStr = (i == values.length - 1) ? valueStr + values[i] : valueStr + values[i] + ","; } params.put(name, valueStr); } try { boolean flag = AlipaySignature.rsaCheckV1(params, alipayConfig.getAlipayPublicKey(), alipayConfig.getCharset(), alipayConfig.getSignType()); if (flag) { logger.info("支付回調信息:{}", params); return "success"; } else { return "error"; } } catch (AlipayApiException e) { logger.error("支付寶錯誤回調:{}", e.getErrMsg()); e.printStackTrace(); return "error"; } } }
上面的 controller 寫瞭兩個接口一個用來 app端的調用,一個給支付用來回調。回調接口的地址要放到剛才配置中的 notifyUrl
屬性裡。
- 由於支付寶回調要使用線上的地址作為回調地址,這裡我推薦兩個解決辦法
- 使用一臺服務器+備案的域名搭建上面的後臺地址
- 使用 花生殼 來實現本地內網穿透
我使用的是 花生殼 作為本次的開發環境,啟動 springboot 的服務,配置好花生殼。後臺部分到目前為止已經結束瞭。
前端部分
UniApp 是一個使用Vue.js 開發所有前端應用的框架,開發者編寫一套代碼,可發佈到iOS、Android、Web(響應式)、以及各種小程序(微信/支付寶/百度/頭條/飛書/QQ/快手/釘釘/淘寶)、快應用等多個平臺。
牛啊,UniApp 能打這麼多的端呢,不過這次我隻使用 APP 端的功能。
在 HBuilder X 中新建一個項目,我目前使用的版本 3.3.10.20220124
創建好項目之後,在 manifest.json 中勾選以下內容
在 index.vue 添加支付相關功能
<template> <view class="content"> <image class="logo" src="/static/logo.png"></image> <view class="text-area"> <text class="title">{{title}}</text> </view> <button type="default" @click="goPay()">點我前去支付</button> </view> </template> <script> export default { data() { return { title: 'Hello' } }, onLoad() { }, methods: { goPay() { uni.request({ url: "https://4789j06630.wocp.fun/pay", success(res) { uni.requestPayment({ provider: 'alipay', orderInfo: res.data.data, success(r) { uni.showModal({ content: "支付成功", showCancel: false }) }, fail(e) { uni.showModal({ content: "支付失敗,原因為: " + e.errMsg, showCancel: false }) }, complete: () => { console.log("payment結束") } }) } }) } } } </script> <style> page{ background-color: #ff5500; } .content { display: flex; flex-direction: column; align-items: center; justify-content: center; } .logo { height: 200rpx; width: 200rpx; margin-top: 200rpx; margin-left: auto; margin-right: auto; margin-bottom: 50rpx; } .text-area { display: flex; justify-content: center; } .title { font-size: 36rpx; color: #8f8f94; } </style>
點擊 點我前去支付 按鈕就可以打開支付寶進行支付瞭。查看接口返回的 log 信息
支付成功圖展示
生成預支付信息
支付成功回調
要記住裡面的 trade_no
一會退款還需要這個。
以上就是支付寶支付的內容瞭。下面開始實現退款功能。
支付寶退款開發
後端部分
在剛才的 BizAlipayService.java 添加以下代碼
/** * 退款 * * @param tradeNo * @param totalAmount * @return */ public AlipayTradeRefundResponse refund(String tradeNo, String totalAmount) { try { DefaultAlipayClient client = client(); AlipayTradeRefundModel alipayTradeRefundModel = new AlipayTradeRefundModel(); alipayTradeRefundModel.setTradeNo(tradeNo); alipayTradeRefundModel.setRefundAmount(totalAmount); AlipayTradeRefundRequest request = new AlipayTradeRefundRequest(); request.setBizModel(alipayTradeRefundModel); AlipayTradeRefundResponse response = client.execute(request); return response; } catch (AlipayApiException e) { logger.error("退款出現問題,詳情:{}", e.getErrMsg()); e.printStackTrace(); } return null; }
在 AlipayController.java 中添加一個接口用於退款操作
/** * 訂單退款 * * @return * @TODO 僅實現瞭全部退款 */ @RequestMapping(value = "/order_refund", method = RequestMethod.GET) public AlipayTradeRefundResponse orderRefund() { AlipayTradeRefundResponse refund = alipayService.refund("2022020922001434041429269213", "0.01"); return refund; }
用的是剛才支付寶回調返回的 trade_no
和 total_amount
。
重啟服務調用接口測試是否可以退款,這裡我為瞭省事使用瞭 Postman 。
手機此時已經收到退款已到賬的消息通知,退款還是很簡單的。後臺也返回瞭響應的支付寶回調
整套支付流程都上傳到 github 瞭可以查看 github的源碼 https://github.com/runbrick/pay_spring
到此這篇關於UniApp+SpringBoot實現支付寶支付和退款的文章就介紹到這瞭,更多相關UniApp SpringBoot 支付寶支付和退款內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- SpringBoot實現簡易支付寶網頁支付功能
- Java 策略模式與模板方法模式相關總結
- java對接支付寶支付項目的實戰記錄
- springboot日志切面通用類實例詳解
- java實現支付寶支付接口的調用