一:項(xiàng)目背景
最近和朋友一起做了一款A(yù)PP ,涉及到支付寶支付, 簡單記錄一下支付寶支付服務(wù)端流程
初涉支付寶支付相對于微信簡單的多,這里簡單體現(xiàn)在支付寶有完整的客服體系。每一步遇到的問題都能為你解答,提供沙箱測試環(huán)境,方便測試
二: 準(zhǔn)備工作
沙箱環(huán)境中有客服入口,再輸入框多輸入幾次人工客服,這個(gè)很重要,期間遇見問題都可以資訊
https://openhome.alipay.com/platform/appDaily.htm?tab=info&ant_source=antsupport

客服入口
開發(fā)文檔 :
https://docs.open.alipay.com/204
賬戶注冊 :
請咨詢客服,百試不爽
支付寶支付接口調(diào)試
常量類
/**
* created by wangch on 2019/8/7
*/
public class AlipayConfig {
/**
* 正式環(huán)境請求地址
*/
public static String ALIPAY_URL = "https://openapi.alipay.com/gateway.do";
/**
* 支付寶分配給開發(fā)者的應(yīng)用ID
*
*/
public static String APP_ID = "";
/**
* 支付接口名稱
*/
public static String PAY_METHOD = "";
/**
* 提現(xiàn)接口名稱
*/
public static String TRANSFER_METHOD = "";
/**
* 僅支持JSON
*/
public static String FORMAT = "JSON";
/**
* 請求使用的編碼格式
*/
public static String CHARSET = "utf-8";
/**
* 商戶生成簽名字符串所使用的簽名算法類型
*/
public static String SIGN_TYPE = "RSA2";
/**
* 支付寶公鑰
*/
public static String ALIPAY_PUBLIC_KEY = "";
/**
* 商戶私鑰
*/
public static String PRIVATE_KEY =" ";
/**
* 回調(diào)地址
*/
public static String CALLBACK_URL = "http://ip:80/api/alipay/getAlipayNotify";
}
接口類
/**
* created by wangch on 2019/8/7
*/
@Api("支付寶支付服務(wù)端接口")
@RestController
@RequestMapping(value = "/alipay")
public class AlipayController {
@Autowired
private IOrderService iOrderService;
@PostMapping("/createPayOrder")
@ApiOperation("創(chuàng)建支付")
public R createPayOrder(@RequestBody Order order) throws Exception{
// 實(shí)例化客戶端
AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfig.ALIPAY_URL, AlipayConfig.APP_ID,
AlipayConfig.PRIVATE_KEY, AlipayConfig.FORMAT, AlipayConfig.CHARSET, AlipayConfig.ALIPAY_PUBLIC_KEY,
AlipayConfig.SIGN_TYPE);
// 實(shí)例化具體API對應(yīng)的request類,類名稱和接口名稱對應(yīng),當(dāng)前調(diào)用接口名稱:alipay.trade.app.pay
AlipayTradeAppPayRequest request = new AlipayTradeAppPayRequest();
// SDK已經(jīng)封裝掉了公共參數(shù),這里只需要傳入業(yè)務(wù)參數(shù)。以下方法為sdk的model入?yún)⒎绞?model和biz_content同時(shí)存在的情況下取biz_content)。
AlipayTradeAppPayModel model = new AlipayTradeAppPayModel();
model.setBody("訂單支付");
model.setSubject("周文斌餐館訂單");
model.setOutTradeNo(order.getCode());// outtradeno 生存訂單
model.setTimeoutExpress("60m");
model.setTotalAmount(order.getPrice() +"");
model.setProductCode("QUICK_MSECURITY_PAY"); // ProductCode這是固定值,參考對應(yīng)的接口文檔的示例值
request.setBizModel(model);
request.setNotifyUrl(AlipayConfig.CALLBACK_URL);//異步回調(diào)url
// 這里和普通的接口調(diào)用不同,使用的是sdkExecute
AlipayTradeAppPayResponse alipayTradeAppPayResponse = alipayClient.sdkExecute(request);
//System.out.println(alipayTradeAppPayResponse.getBody());//就是orderString 可以直接給客戶端請求,無需再做處理。
return R.ok().put("orderStr",alipayTradeAppPayResponse.getBody());
}
@PostMapping("/getAlipayNotify")
@ApiOperation("異步回調(diào)")
public R getAlipayNotify(HttpServletRequest request) {
R r = new R();
Map<String, String> params = new HashMap<String, String>();
Map requestParams = request.getParameterMap();
for (Iterator 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] + ",";
}
// 亂碼解決,這段代碼在出現(xiàn)亂碼時(shí)使用。
// valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");
params.put(name, valueStr);
// 獲取支付訂單號(hào)
String trade_no = params.get("trade_no");
// 將訂單號(hào)存入訂單表中
String out_trade_no = params.get("out_trade_no");
iOrderService.insertTradeNo(trade_no,out_trade_no);
}
// 切記alipaypublickey是支付寶的公鑰,請去open.alipay.com對應(yīng)應(yīng)用下查看。
try {
boolean flag = AlipaySignature.rsaCheckV1(params, AlipayConfig.ALIPAY_PUBLIC_KEY, AlipayConfig.CHARSET,
AlipayConfig.SIGN_TYPE);
if (flag) {
String trade_status = params.get("trade_status");
String out_trade_no = params.get("out_trade_no");
if ("TRADE_SUCCESS".equals(trade_status)) { // 交易支付成功的執(zhí)行相關(guān)業(yè)務(wù)邏輯
// 支付成功
int count = iOrderService.updateOrderPayByOrderCode(out_trade_no,1);
if (count > 0){
r.put("msg","支付成功,修改訂單狀態(tài)成功");
} else {
r.put("msg","支付成功,修改訂單狀態(tài)失敗,請聯(lián)系管理員");
}
} else if ("TRADE_CLOSED".equals(trade_status)) { // 未付款交易超時(shí)關(guān)閉,或支付完成后全額退款,執(zhí)行相關(guān)業(yè)務(wù)邏輯
r.error("支付超時(shí),請重試");
}
} else {
r.error("支付失敗,請重試");
}
} catch (Exception e) {
e.printStackTrace();
}
return r;
}
@GetMapping("/payCallback")
@ApiOperation("查詢支付結(jié)果")
public R payCallback(@RequestParam Integer orderId) throws Exception{
R r = new R();
// 如果當(dāng)前訂單支付結(jié)果已經(jīng) 修改過 次直接返回 收到的支付結(jié)果
Order order = iOrderService.getOrderInfo(orderId);
if(order.getPayStatus() == 1) {
r.put("code",0);
r.put("msg","訂單支付成功");
return r;
}
Map<String, String> payStatus = iOrderService.getOrderCodeAndAliTradeCode(orderId);
String out_trade_no = payStatus.get("code");
String trade_no = payStatus.get("trade_no");
AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfig.ALIPAY_URL,AlipayConfig.APP_ID,AlipayConfig.PRIVATE_KEY,
AlipayConfig.FORMAT,AlipayConfig.CHARSET,AlipayConfig.ALIPAY_PUBLIC_KEY,AlipayConfig.SIGN_TYPE);
AlipayTradeQueryRequest request = new AlipayTradeQueryRequest();
request.setBizContent("{ \"out_trade_no\":\""+out_trade_no+"\"}");
AlipayTradeQueryResponse response = alipayClient.execute(request);
//保存訂單號(hào) 和支付寶 交易號(hào)的關(guān)系
iOrderService.insertTradeNo(response.getTradeNo(),response.getOutTradeNo());
if(response.getTradeStatus().equals("TRADE_SUCCESS")){
r.put("code",1);
r.put("msg","訂單支付成功");
System.out.println("訂單支付成功");
iOrderService.updateOrderPayByOrderCode(response.getOutTradeNo(),1);
iOrderService.pushOrderPaySuccessNotifications(orderId);
return r;
};
if(response.getTradeStatus().equals("WAIT_BUYER_PAY")){
r.put("code",0);
r.put("msg","等待付款");
iOrderService.updateOrderPayByOrderCode(response.getOutTradeNo(),0);
System.out.println("等待付款");
return r;
};
return r;
}
}
四:總結(jié)
支付寶支付對初學(xué)者相對友好,遇到問題記得問客服,切記