支付寶支付完整案例
阿新 • • 發佈:2019-01-12
配置 -a orm 輸入 exceptio clas rac 完成後 瀏覽器
一: 開發前的準備
任何第三方對接都要獲取去平臺註冊獲取appid、密鑰等信息。(QQ登錄,微信登錄,微信支付,支付寶支付......等等)
支付寶商戶信息申請:https://openhome.alipay.com/platform/appDaily.htm?tab=info
二:支付寶支付案例
支付寶核心代碼開發(參考官方PC支付下單文檔 https://docs.open.alipay.com/270/alipay.trade.page.pay)
/** * 支付寶支付 * @param price商品價格 */ public String pay(String price,String orderId) { log.info("調用支付寶支付接口開始,price:{}"+price); //1. 獲得初始化的AlipayClient AlipayClient alipayClient = new DefaultAlipayClient(aliPayConfig.getGatewayUrl(), aliPayConfig.getAppId(), aliPayConfig.getMerchantPrivateKey(),"json", aliPayConfig.getCharSet(), aliPayConfig.getAlipayPublicKey(), aliPayConfig.getSignType()); //2. 設置請求參數 AlipayTradePagePayRequest alipayRequest = newAlipayTradePagePayRequest(); alipayRequest.setReturnUrl(aliPayConfig.getReturnUrl());//同步回調地址 alipayRequest.setNotifyUrl(aliPayConfig.getNotifyUrl());//異步回調地址 String out_trade_no = orderId;//商戶訂單號,商戶網站訂單系統中唯一訂單號 String total_amount = price; //付款金額 String subject = "支付寶下單測試"; //訂單名稱 String body = "這是一條商品描述記錄";//商品描述,可為空 //3. 將請求參數拼接成json格式。 alipayRequest.setBizContent("{\"out_trade_no\":\""+ out_trade_no +"\"," + "\"total_amount\":\""+ total_amount +"\"," + "\"subject\":\""+ subject +"\"," + "\"body\":\""+ body +"\"," + "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}"); //若想給BizContent增加其他可選請求參數,以增加自定義超時時間參數timeout_express來舉例說明 //alipayRequest.setBizContent("{\"out_trade_no\":\""+ out_trade_no +"\"," // + "\"total_amount\":\""+ total_amount +"\"," // + "\"subject\":\""+ subject +"\"," // + "\"body\":\""+ body +"\"," // + "\"timeout_express\":\"10m\"," // + "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}"); //請求參數可查閱【電腦網站支付的API文檔-alipay.trade.page.pay-請求參數】章節 //請求 String result = null; try { result = alipayClient.pageExecute(alipayRequest).getBody(); } catch (AlipayApiException e) { e.printStackTrace(); } log.info("調用支付寶支付接口完成"); return result; }
三:用戶登錄並支付
用戶輸入支付密碼後,會調用配置文件裏的異步、同步回調路徑。(這一部分是支付寶自動完成的,參考sdk支付案例源碼http://p.tb.cn/rmsportal_6680_alipay.trade.page.pay-JAVA-UTF-8.zip)
/** * 異步回調notifyUrl */ @RequestMapping("/notifyUrl") public String asynCallBack(HttpServletRequest req) throws UnsupportedEncodingException{ Map<String,String> params = new HashMap<String,String>(); Map<String,String[]> requestParams = req.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); } log.info("支付寶異步回調通知,params:{}"+params); // 調用SDK驗證簽名 boolean signVerified = this.signVerified(params); if(signVerified) {//驗證成功 //商戶訂單號 String out_trade_no = new String(req.getParameter("out_trade_no").getBytes("ISO-8859-1"),"UTF-8"); //支付寶交易號 String trade_no = new String(req.getParameter("trade_no").getBytes("ISO-8859-1"),"UTF-8"); //交易狀態 String trade_status = new String(req.getParameter("trade_status").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)在商戶網站的訂單系統中查到該筆訂單的詳細,並執行商戶的業務程序 //如果有做過處理,不執行商戶的業務程序 //註意: //付款完成後,支付寶系統發送該交易狀態通知 } return "success"; }else {//驗證失敗 return "fail"; } } /** * 同步回調 */ @RequestMapping("/returnUrl") @ResponseBody public void synCallBack(HttpServletRequest req, HttpServletResponse resp) throws Exception { resp.setContentType("text/html;charset=utf-8"); PrintWriter writer = resp.getWriter(); // 拼裝請求參數 Map<String,String> params = new HashMap<String,String>(); Map<String,String[]> requestParams = req.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); } log.info("支付寶同步回調通知,params:{}"+params); // 調用SDK驗證簽名 boolean signVerified = this.signVerified(params); if(!signVerified) { log.info("SDK驗簽失敗"); return; } //商戶訂單號 String out_trade_no = params.get("out_trade_no"); //支付寶交易號 String trade_no = params.get("trade_no"); //付款金額 String total_amount = params.get("total_amount"); // 封裝成html 瀏覽器模擬去提交 String htmlFrom="<form name=‘punchout_form‘ " + "method=‘post‘ " + "action=‘http://127.0.0.1:8081/pay/synSuccessPage‘ >" + "<input type=‘hidden‘ name=‘outTradeNo‘ value=‘"+out_trade_no+"‘>" + "<input type=‘hidden‘ name=‘tradeNo‘ value=‘"+trade_no+"‘>" + "<input type=‘hidden‘ name=‘totalAmount‘ value=‘"+total_amount+"‘>" + "<input type=‘submit‘ value=‘立即支付‘ style=‘display:none‘>" + "</form>" + "<script>document.forms[0].submit();</script>"; writer.println(htmlFrom); } // 支付寶支付成功頁面回調是get方式,會暴露參數。所以我們這裏來以post表單隱藏參數 @RequestMapping(value = "/synSuccessPage", method = RequestMethod.POST) @ResponseBody private String synSuccessPage(HttpServletRequest request, String outTradeNo, String tradeNo, String totalAmount) { request.setAttribute("outTradeNo", outTradeNo); request.setAttribute("tradeNo", tradeNo); request.setAttribute("totalAmount", totalAmount); return "您的訂單號: "+outTradeNo+"已經支付成功, 支付金額: "+totalAmount+" 元"; } /** * 調用SDK驗證簽名 * @param params 支付寶回調參數 */ public boolean signVerified(Map<String,String> params) { try { log.info("調用SDK驗證簽名...."); return AlipaySignature.rsaCheckV1(params, aliPayConfig.getAlipayPublicKey(), aliPayConfig.getCharSet(), aliPayConfig.getSignType()); } catch (AlipayApiException e) { return false; } }
四:成功頁面
支付寶支付完整案例