一.支付寶測試環境開發的前期准備
1.下載Java支付的demo
demo下載地址:https://docs.open.alipay.com/270/106291/
1.下載解壓導入idea
readme.txt請好好看一下,里邊是關於一些下載demo到jsp頁面實現的過程,供學習參考,解壓的demo中只有一個Java配置類,其余都是JSP頁面。
3. 配置AlipayConfig
(1).免費注冊螞蟻金服開發者賬號
注冊地址:https://open.alipay.com ,用你的支付寶賬號掃碼或者賬號登錄,完善個人信息,選擇服務類型(我選的是自研)。
(2).設置app_id和gatewayUrl(支付寶網關)
其中密鑰需要自己生成,appID和支付寶網關是已經給好的,網關有dev字樣,表明是用於開發測試。
(1).設置密匙
點擊“生成方法”,打開界面如下圖:
下載密鑰生成工具,解壓打開后,雙擊腳本文件 “RSA簽名驗簽工具.bat” 即運行RSA簽名驗簽工具,選擇PKCS8(Java適用)和2048位生成密鑰:
如果沒有設置過,此時顯示文本是“設置應用公鑰”,我這里是已經設置過得,設置過得可以永久使用了。
設置方法,“打開密鑰文件路徑”:
復制應用公鑰2048.txt中的內容到點擊“設置應用公鑰”的彈出框中,保存:
-
商戶私鑰(merchant_private_key)
復制 應用私鑰2048.txt 中的內容到merchant_private_key中。
-
支付寶公鑰(alipay_public_key)
點擊如上圖鏈接,復制彈出框里面的內容到alipay_public_key。
如果這個設置不對,結果是:支付成功,但是驗簽失敗。
如果是正式環境,需要上傳到對應的應用中:
(4).服務器異步通知頁面路徑(notify_url)
如果沒有改名,修改IP和端口號就可以了,我自己的如下:
http://localhost:8080/alipay/
alipayNotifyNotice(5).頁面跳轉同步通知頁面路徑(return_url)
http://localhost:8080/alipay/
alipayReturnNotice
二、將支付寶支付整合到springboot框架中
1. 項目架構
- 項目架構:springboot
- 數據庫:mysql
- 部署環境:tomcat8.0
- 開發環境:jdk8、idea
- 支付:支付寶、微信
整合到springboot一樣,我們需要像沙箱測試環境一樣,需要修改支付的配置信息
2、數據庫代碼
主要包括以下的數據庫表:
- user:用戶表
- order:支付產生的訂單
- flow:流水賬
- product:商品表:用於模擬購買商品。
CREATE TABLE `user` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '用戶Id',
`username` varchar(128) DEFAULT NULL COMMENT '用戶名',
`sex` varchar(20) DEFAULT NULL COMMENT '性別',
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=15 DEFAULT CHARSET=utf8 COMMENT='用戶表';
CREATE TABLE `flow` (
`id` varchar(20) NOT NULL,
`flow_no` varchar(32) DEFAULT NULL COMMENT '流水號',
`order_no` varchar(20) DEFAULT NULL COMMENT '訂單號',
`product_id` varchar(20) DEFAULT NULL COMMENT '產品主鍵ID',
`pay_amount` varchar(11) DEFAULT NULL COMMENT '支付金額',
`pay_type` int(11) DEFAULT NULL COMMENT '支付方式\r\n 1:支付寶\r\n 2:微信',
`buy_count` int(11) DEFAULT NULL COMMENT '購買個數',
`create_time` datetime DEFAULT NULL COMMENT '創建時間',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='流水表';
CREATE TABLE `orders` (
`id` varchar(20) NOT NULL,
`order_no` varchar(20) DEFAULT NULL COMMENT '訂單號',
`order_status` varchar(20) DEFAULT NULL COMMENT '訂單狀態\r\n 10:待付款\r\n 20:已付款',
`order_amount` varchar(11) DEFAULT NULL COMMENT '訂單金額',
`pay_amount` varchar(11) DEFAULT NULL COMMENT '實際支付金額',
`product_id` varchar(20) DEFAULT NULL COMMENT '產品表外鍵ID',
`buy_count` int(11) DEFAULT NULL COMMENT '產品購買的個數',
`create_time` datetime DEFAULT NULL COMMENT '訂單創建時間',
`pay_time` datetime DEFAULT NULL COMMENT '支付時間',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='訂單表';
CREATE TABLE `product` (
`id` varchar(20) NOT NULL,
`name` varchar(20) DEFAULT NULL COMMENT '產品名稱',
`price` varchar(11) DEFAULT NULL COMMENT '價格',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='產品表 ';
3、dao數據接口層
可以使用通用mapper
,或者逆向工程
就行。以訂單order為例給出:
package com.suncy.alipay.dao;
import com.suncy.alipay.model.Order;
import org.apache.ibatis.annotations.Mapper;
import org.springframework.stereotype.Component;
/**
* 訂單
* @author scy 2019/3/6
*/
@Mapper
@Component
public interface OrdersMapper {
int insert(Order record);
Order findByOrderId(String id);
int updateByPrimaryKeySelective(Order record);
}
4、service層
以訂單order為例給出:
package com.suncy.alipay.service;
import com.suncy.alipay.model.Product;
import org.springframework.stereotype.Service;
import java.util.List;
/**
* 產品操作 service
* @author scy 2019/3/6
*/
@Service
public interface ProductService {
/**
* 獲取所有產品列表
* @return
*/
List<Product> getProducts();
/**
* 根據產品ID查詢產品詳情
* @param productId
* @return
*/
Product getProductById(String productId);
}
4、支付寶支付controller(支付流程)
支付流程圖
首先,啟動項目后,輸入http://localhost:8080/alipay/product,會進入到商品頁面,如下
下面是頁面代碼
商品頁面(products.jsp)
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8" %>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn" %>
<%
String path = request.getContextPath();
String basePath = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + path + "/";
%>
<script type="text/javascript" src="<%=basePath%>js/jquery.min.js"></script>
<html>
<head>
<base href="<%=basePath%>">
<title>產品列表</title>
<link rel="stylesheet" type="text/css" href="<%=path%>css/products.css">
</head>
<body>
<div id="product_frame">
<p><img id="image_logo" src="<%=path%>images/head.jpg"></p>
<h3 align="center">產品列表</h3>
<table border="1" cellspacing="0" cellpadding="0">
<tbody>
<tr>
<th>產品編號</th>
<th>產品名稱</th>
<th>產品價格</th>
<th>操作</th>
</tr>
<c:if test="${!empty productList}">
<c:forEach items="${productList}" var="product">
<tr>
<td>${product.id}</td>
<td>${product.name }</td>
<td>${product.price }</td>
<td>
<div id="login_control">
<a href="<%=basePath%>alipay/goConfirm?productId=${product.id }">
<input type="button" id="btn_buyProduct" value="購買">
</a>
</div>
</td>
</tr>
</c:forEach>
</c:if>
</tbody>
</table>
<input type="hidden" id="hdnContextPath" name="hdnContextPath" value="<%=basePath%>"/>
</div>
</body>
</html>
點擊上面的購買,進入到訂單頁面
填寫個數,然后點擊生成訂單,調用如下代碼
/**
* @param order
* @return
* @throws Exception
*/
@Transactional(rollbackFor = Exception.class)
@RequestMapping(value = "createOrder")
public ModelAndView createOrder(Order order,Integer buyCounts) throws Exception {
Product p = productService.getProductById(order.getProductId());
String orderId = Join.creatOrder("DDH");
order.setId(orderId);
order.setOrderNo(orderId);
order.setCreateTime(new Date());
order.setBuyCount(buyCounts);
order.setOrderAmount(String.valueOf(Float.valueOf(p.getPrice()) * order.getBuyCount()));
order.setOrderStatus(OrderStatusEnum.WAIT_PAY.key);
orderService.saveOrder(order);
ModelAndView mv = new ModelAndView("goPay");
mv.addObject("order", order);
mv.addObject("p", p);
return mv;
}
進入到選擇支付頁面
然后,我們選擇支付寶支付,進入到了我們支付的頁面了。
調用了如下代碼:
/**
* @Description: 前往支付寶第三方網關進行支付
*/
@RequestMapping(value = "goAlipay", produces = "text/html; charset=UTF-8")
@ResponseBody
public String goAlipay(String orderId) throws Exception {
Order order = orderService.getOrderById(orderId);
Product product = productService.getProductById(order.getProductId());
//獲得初始化的AlipayClient
AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfig.gatewayUrl, AlipayConfig.app_id, AlipayConfig.merchant_private_key, "json", AlipayConfig.charset, AlipayConfig.alipay_public_key, AlipayConfig.sign_type);
//設置請求參數
AlipayTradePagePayRequest alipayRequest = new AlipayTradePagePayRequest();
alipayRequest.setReturnUrl(AlipayConfig.return_url);
alipayRequest.setNotifyUrl(AlipayConfig.notify_url);
//商戶訂單號,商戶網站訂單系統中唯一訂單號,必填
String out_trade_no = orderId;
//付款金額,必填
String total_amount = order.getOrderAmount();
//訂單名稱,必填
String subject = product.getName();
//商品描述,可空
String body = "用戶訂購商品個數:" + order.getBuyCount();
// 該筆訂單允許的最晚付款時間,逾期將關閉交易。取值范圍:1m~15d。m-分鍾,h-小時,d-天,1c-當天(1c-當天的情況下,無論交易何時創建,都在0點關閉)。 該參數數值不接受小數點, 如 1.5h,可轉換為 90m。
String timeout_express = "1c";
alipayRequest.setBizContent("{\"out_trade_no\":\""+ out_trade_no +"\","
+ "\"total_amount\":\""+ total_amount +"\","
+ "\"subject\":\""+ subject +"\","
+ "\"body\":\""+ body +"\","
+ "\"timeout_express\":\""+ timeout_express +"\","
+ "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}");
//請求
String result = alipayClient.pageExecute(alipayRequest).getBody();
return result;
}
/**
* @Description: 支付寶同步通知頁面
*/
@RequestMapping(value = "alipayReturnNotice")
public ModelAndView alipayReturnNotice(HttpServletRequest request, HttpServletRequest response) throws Exception {
log.info("支付成功, 進入同步通知接口...");
//獲取支付寶GET過來反饋信息
Map<String,String> params = new HashMap<String,String>();
Map<String,String[]> requestParams = request.getParameterMap();
for (Iterator<String> iter = requestParams.keySet().iterator(); iter.hasNext();) {
String name = (String) iter.next();
String[] values = (String[]) requestParams.get(name);
String valueStr = "";
for (int i = 0; i < values.length; i++) {
valueStr = (i == values.length - 1) ? valueStr + values[i]
: valueStr + values[i] + ",";
}
//亂碼解決,這段代碼在出現亂碼時使用
valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");
params.put(name, valueStr);
}
//調用SDK驗證簽名
boolean signVerified = AlipaySignature.rsaCheckV1(params, AlipayConfig.alipay_public_key, AlipayConfig.charset, AlipayConfig.sign_type);
ModelAndView mv = new ModelAndView("alipaySuccess");
//——請在這里編寫您的程序(以下代碼僅作參考)——
if(signVerified) {
//商戶訂單號
String out_trade_no = new String(request.getParameter("out_trade_no").getBytes("ISO-8859-1"),"UTF-8");
//支付寶交易號
String trade_no = new String(request.getParameter("trade_no").getBytes("ISO-8859-1"),"UTF-8");
//付款金額
String total_amount = new String(request.getParameter("total_amount").getBytes("ISO-8859-1"),"UTF-8");
// 修改訂單狀態為支付成功,已付款; 同時新增支付流水
orderService.updateOrderStatus(out_trade_no, trade_no, total_amount);
Order order = orderService.getOrderById(out_trade_no);
Product product = productService.getProductById(order.getProductId());
log.info("********************** 支付成功(支付寶同步通知) **********************");
log.info("* 訂單號: {}", out_trade_no);
log.info("* 支付寶交易號: {}", trade_no);
log.info("* 實付金額: {}", total_amount);
log.info("* 購買產品: {}", product.getName());
log.info("***************************************************************");
mv.addObject("out_trade_no", out_trade_no);
mv.addObject("trade_no", trade_no);
mv.addObject("total_amount", total_amount);
mv.addObject("productName", product.getName());
}else {
log.info("支付, 驗簽失敗...");
}
return mv;
}
/**
* @Description: 支付寶異步 通知頁面
*/
@RequestMapping(value = "alipayNotifyNotice")
@ResponseBody
public String alipayNotifyNotice(HttpServletRequest request, HttpServletRequest response) throws Exception {
log.info("支付成功, 進入異步通知接口...");
//獲取支付寶POST過來反饋信息
Map<String,String> params = new HashMap<String,String>();
Map<String,String[]> requestParams = request.getParameterMap();
for (Iterator<String> iter = requestParams.keySet().iterator(); iter.hasNext();) {
String name = (String) iter.next();
String[] values = (String[]) requestParams.get(name);
String valueStr = "";
for (int i = 0; i < values.length; i++) {
valueStr = (i == values.length - 1) ? valueStr + values[i]
: valueStr + values[i] + ",";
}
//亂碼解決,這段代碼在出現亂碼時使用
/*valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");*/
params.put(name, valueStr);
}
//調用SDK驗證簽名
boolean signVerified = AlipaySignature.rsaCheckV1(params, AlipayConfig.alipay_public_key, AlipayConfig.charset, AlipayConfig.sign_type);
//——請在這里編寫您的程序(以下代碼僅作參考)——
/* 實際驗證過程建議商戶務必添加以下校驗:
1、需要驗證該通知數據中的out_trade_no是否為商戶系統中創建的訂單號,
2、判斷total_amount是否確實為該訂單的實際金額(即商戶訂單創建時的金額),
3、校驗通知中的seller_id(或者seller_email) 是否為out_trade_no這筆單據的對應的操作方(有的時候,一個商戶可能有多個seller_id/seller_email)
4、驗證app_id是否為該商戶本身。
*/
//驗證成功
if(signVerified) {
//商戶訂單號
String out_trade_no = new String(request.getParameter("out_trade_no").getBytes("ISO-8859-1"),"UTF-8");
//支付寶交易號
String trade_no = new String(request.getParameter("trade_no").getBytes("ISO-8859-1"),"UTF-8");
//交易狀態
String trade_status = new String(request.getParameter("trade_status").getBytes("ISO-8859-1"),"UTF-8");
//付款金額
String total_amount = new String(request.getParameter("total_amount").getBytes("ISO-8859-1"),"UTF-8");
if(trade_status.equals("TRADE_FINISHED")){
//判斷該筆訂單是否在商戶網站中已經做過處理
//如果沒有做過處理,根據訂單號(out_trade_no)在商戶網站的訂單系統中查到該筆訂單的詳細,並執行商戶的業務程序
//如果有做過處理,不執行商戶的業務程序
//注意: 尚自習的訂單沒有退款功能, 這個條件判斷是進不來的, 所以此處不必寫代碼
//退款日期超過可退款期限后(如三個月可退款),支付寶系統發送該交易狀態通知
}else if (trade_status.equals("TRADE_SUCCESS")){
//判斷該筆訂單是否在商戶網站中已經做過處理
//如果沒有做過處理,根據訂單號(out_trade_no)在商戶網站的訂單系統中查到該筆訂單的詳細,並執行商戶的業務程序
//如果有做過處理,不執行商戶的業務程序
//注意:
//付款完成后,支付寶系統發送該交易狀態通知
// 修改叮當狀態,改為 支付成功,已付款; 同時新增支付流水
orderService.updateOrderStatus(out_trade_no, trade_no, total_amount);
Order order = orderService.getOrderById(out_trade_no);
Product product = productService.getProductById(order.getProductId());
log.info("********************** 支付成功(支付寶異步通知) **********************");
log.info("* 訂單號: {}", out_trade_no);
log.info("* 支付寶交易號: {}", trade_no);
log.info("* 實付金額: {}", total_amount);
log.info("* 購買產品: {}", product.getName());
log.info("***************************************************************");
}
log.info("支付成功...");
}else {//驗證失敗
log.info("支付, 驗簽失敗...");
}
return "success";
}
這段代碼都可以在阿里支付的demo里面找到的,只需要復制過來,然后改改,整合到springboot環境即可。
支付成功后返回支付成功信息:
數據庫已經有數據,支付狀態已經更改
上面就是將阿里支付寶支付整合到springboot的全過程了,如果還有什么疑問,可以留言或者私信我,有問必答,源碼私聊,剛做好的項目,暫時沒有上傳到GitHub。