想熟悉支付寶接口支付,後面可能會用,不如在課設中試試手。好吧據說支付寶不微信支付要簡單些,就拿支付寶的先練下手吧。html
基本學習流程,百度一下,找篇博客看下。java
推薦下面這個篇博客,講的挺好的,複製過來。git
固然在最後我會把我遇到的問題寫出來,避免後人在走彎路,並將支付寶的demo整合到個人ssm項目中,同時,會詳細講下如何將demo中的jsp跟據需求改成contral層。github
比較能夠的操做流程:別人寫的很好了,我就不想寫了:https://github.com/OUYANGSIHAI/sihai-maven-ssm-alipayexpress
下載地址:https://docs.open.alipay.com/270/106291/json
readme.txt請好好看一下。服務器
只有一個Java配置類,其他都是JSP。微信
(1).註冊螞蟻金服開發者帳號(免費,不像蘋果會收取費用)app
註冊地址:https://open.alipay.com ,用你的支付寶帳號掃碼登陸,完善我的信息,選擇服務類型(我選的是自研)。框架
(2).設置app_id和gatewayUrl
其中密鑰須要本身生成,appID和支付寶網關是已經給好的,網關有dev字樣,代表是用於開發測試。
(3).設置密鑰
點擊「生成方法」,打開界面以下:
下週密鑰生成工具,解壓打開後,選擇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.trade.page.pay-JAVA-UTF-8/notify_url.jsp
(5).頁面跳轉同步通知頁面路徑(return_url)
http://localhost:8080/alipay.trade.page.pay-JAVA-UTF-8/return_url.jsp
測試用的支付寶買家帳戶能夠在「沙箱帳號」這個頁面能夠找到:
支付成功後,驗簽結果:
好吧,要是你成功跑起來的那麼你很幸運,沒有到什麼坑,我遇到了以下坑點。記錄一下,以供參考:
1.代碼中:
要填寫公鑰,注意這個不是生成器中的公鑰,因此不能從那個裏面或者從txt中粘貼,而要重網站上那個查看公鑰處粘貼:
2. 支付寶網關要改,注意測試環境中是alipaydev:
3. 從官網下載的沙箱版支付寶怎麼登入:
登入的帳號去沙箱帳號中看,注意不是你的支付寶帳號,也不須要從新註冊!!!!
注意打支付寶很人性話的給了咱們許多錢,能夠肆意的花費了,最後會從你的買家餘額到買家餘額中。
好的,基本注意這些,demo是能夠跑起來的了。
下面講下如何加入到ssm框架中,隨意控制呢:
你得先對支付寶的流程有一個大體的瞭解,總體上了有把握,能使你思路清晰,請看圖:
這是支付寶提供的圖片,大體過程都表達出來了。
其實到這裏不由要問,那個同步和異步是幹嗎的呀,我也不知道,百度啊!!!
我找到了,比較新的解釋:
支付寶同步回調和異步回調
當一個支付請求被髮送到支付渠道方,支付渠道會很快返回一個結果。可是這個結果,只是告訴你調用成功了,不是扣款成功,這叫同步調用。
不少新手會拿這個結果 看成支付成功了,那就會被坑死,結果就是支付成功率特別高,伴隨着一堆沒法解釋的壞帳率,測試人員尤爲要注意測試數據的篡改:金額,同步返回結果,訂單號等。
同步請求參數裏面會有一個回調地址,這個地址是支付渠道在扣款成功後調用的,這叫異步調用。
通常同步接口僅檢查參數是否正確,簽名是否無誤等。異步接口才告訴你扣款結果。
通常異步接口有5秒之內的延遲。調用不成功會重試。有時候是這邊成功了,但支付渠道側沒收到返回,因而會繼續調。
當天的支付到次日還在 被異步調用也都是正常的。這也是開發人員須要特別注意的地方,不要當作重複支付。
測試人員也要對重複回調進行測試,應只有一次有效。這還不是最坑的,通常 支付渠道側,只有支付成功了才通知你。
要是支付失敗了,壓根兒都不告訴你。
另外一方面,如何老收不到異步結果呢?那就得查查了。同步結果不可靠,異步調用不可靠,那怎麼肯定支付結果?最終的殺招就是查單了,
反查,通常支付渠道側都 會提供反查接口,定時獲取DB中待支付的訂單調用支付渠道側的反查接口,最終把支付渠道側扣款成功的訂單完成掉。
固然,這個沒有將過多細節,更多內容能夠參考其餘的文檔,大體就是說,同步通知能夠做爲支付成功參考,可是準確得出結論還得靠異步的回覆
好吧,因爲異步必須在服務器上才能測試,因此這裏我暫時是以同步的做爲判斷依據:
開始整合:
思路大體是這樣子:
在一個支付頁面 --》》》 點擊支付 ----》》》 跳轉處處理函數 -----》 在這個函數裏面設置你真正要想支付寶請求的一些參數,以及同步異步回調地址!!! ----》 而後他在封裝好信息後發送給支付寶
------》》》 頁面會彈出一個支付寶二維碼,你掃碼付錢完 -------》》》 支付寶會向你給的同步異步回調地址發送請求(也就是你要寫個接收請求的函數咯,在裏面能夠寫一些你要的邏輯了) ------》》》
根據回調結果,挑戰頁面給出回調結果 ------ 》》》 finish ! ! !!!!
具體的:
1. 支付請求給個post的地址:也就是你處理支付的函數:
2.在contral裏面寫對應 的處理函數裏咯:
函數裏面其實就是把支付寶demo的jsp處理粘過來了:
@ResponseBody
@RequestMapping(value = "/goods/alipay2")
public String alipay(
HttpServletRequest request
) throws AlipayApiException, IOException {
System.out.println("-------------------/goods/alipay2-----------------------");
//得到初始化的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 = new String(request.getParameter("WIDout_trade_no").getBytes("ISO-8859-1"),"UTF-8");
//付款金額,必填
String total_amount = new String(request.getParameter("WIDtotal_amount").getBytes("ISO-8859-1"),"UTF-8");
//訂單名稱,必填
String subject = new String(request.getParameter("WIDsubject").getBytes("ISO-8859-1"),"UTF-8");
//商品描述,可空
String body = new String(request.getParameter("WIDbody").getBytes("ISO-8859-1"),"UTF-8");
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 = alipayClient.pageExecute(alipayRequest).getBody();
//輸出
System.out.print("trade.page:----" + result);
return result;
}
2.處理回調地址:
就是一個請求的處理函數,不清楚他們爲何要在請求後面加一個.action, 反正我沒加。。。
3.寫處理函數咯:也是把支付寶的jsp中對應的一個粘過來,
/* *
* 功能:支付寶服務器同步通知頁面
* 日期:2017-03-30
* 說明:
* 如下代碼只是爲了方便商戶測試而提供的樣例代碼,商戶能夠根據本身網站的須要,按照技術文檔編寫,並不是必定要使用該代碼。
* 該代碼僅供學習和研究支付寶接口使用,只是提供一個參考。
*************************頁面功能說明*************************
* 該頁面僅作頁面展現,業務邏輯處理請勿在該頁面執行
*/
@RequestMapping(value = "/goods/alipayReturnNotice")
public ModelAndView alipayReturnNotice(
HttpServletRequest request,
HttpServletRequest response)
throws Exception {
System.out.println("-------------------alipayReturnNotice--------------------------");
//獲取支付寶GET過來反饋信息
//獲取支付寶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);
}
boolean signVerified = AlipaySignature.rsaCheckV1(params, AlipayConfig.alipay_public_key, AlipayConfig.charset, AlipayConfig.sign_type); //調用SDK驗證簽名
ModelAndView mv = new ModelAndView("/goods/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");
mv.addObject("out_trade_no", out_trade_no);
mv.addObject("trade_no", trade_no);
mv.addObject("total_amount", total_amount);
mv.addObject("productName", "name");
System.out.println("驗籤成功");
System.out.println("trade_no:"+trade_no+" out_trade_no:"+out_trade_no+" total_amount:"+total_amount);
// out.println("trade_no:"+trade_no+"<br/>out_trade_no:"+out_trade_no+"<br/>total_amount:"+total_amount);
}else {
// out.println("驗籤失敗");
System.out.println("驗籤失敗");
}
//——請在這裏編寫您的程序(以上代碼僅做參考)——
return mv;
}
4.
/**
*
* @Title: AlipayController.java
* @Package com.sihai.controller
* @Description: 支付寶異步 通知頁面
* Copyright: Copyright (c) 2017
* Company:FURUIBOKE.SCIENCE.AND.TECHNOLOGY
*
* @author sihai
* @date 2017年8月23日 下午8:51:13
* @version V1.0
*/
@RequestMapping(value = "/goods/alipayNotifyNotice")
@ResponseBody
public String alipayNotifyNotice(
HttpServletRequest request,
HttpServletRequest response)
throws Exception {
/*************************頁面功能說明*************************
* 建立該頁面文件時,請留心該頁面文件中無任何HTML代碼及空格。
* 該頁面不能在本機電腦測試,請到服務器上作測試。請確保外部能夠訪問該頁面。
* 若是沒有收到該頁面返回的 success
* 建議該頁面只作支付成功的業務邏輯處理,退款的處理請以調用退款查詢接口的結果爲準。
*/
System.out.println("-----------------alipayNotifyNotice--------------------------");
//獲取支付寶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);
}
boolean signVerified = AlipaySignature.rsaCheckV1(params, AlipayConfig.alipay_public_key, AlipayConfig.charset, AlipayConfig.sign_type); //調用SDK驗證簽名
//——請在這裏編寫您的程序(如下代碼僅做參考)——
/* 實際驗證過程建議商戶務必添加如下校驗:
一、須要驗證該通知數據中的out_trade_no是否爲商戶系統中建立的訂單號,
二、判斷total_amount是否確實爲該訂單的實際金額(即商戶訂單建立時的金額),
三、校驗通知中的seller_id(或者seller_email) 是否爲out_trade_no這筆單據的對應的操做方(有的時候,一個商戶可能有多個seller_id/seller_email)
四、驗證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");
if(trade_status.equals("TRADE_FINISHED")){
//判斷該筆訂單是否在商戶網站中已經作過處理
//若是沒有作過處理,根據訂單號(out_trade_no)在商戶網站的訂單系統中查到該筆訂單的詳細,並執行商戶的業務程序
//若是有作過處理,不執行商戶的業務程序
//注意:
//退款日期超過可退款期限後(如三個月可退款),支付寶系統發送該交易狀態通知
}else if (trade_status.equals("TRADE_SUCCESS")){
//判斷該筆訂單是否在商戶網站中已經作過處理
//若是沒有作過處理,根據訂單號(out_trade_no)在商戶網站的訂單系統中查到該筆訂單的詳細,並執行商戶的業務程序
//若是有作過處理,不執行商戶的業務程序
//注意:
//付款完成後,支付寶系統發送該交易狀態通知
}
System.out.println("success");
}else {//驗證失敗
System.out.println("fail");
//調試用,寫文本函數記錄程序運行狀況是否正常
//String sWord = AlipaySignature.getSignCheckContentV1(params);
//AlipayConfig.logResult(sWord);
}
//——請在這裏編寫您的程序(以上代碼僅做參考)——
return "success";
}
5.能夠了給測試的jsp,和從成功的jsp吧:
alipay.jsp:
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<form name=alipayment action= "${pwd }/goods/alipay2" method=post
target="_blank">
<div id="body1" class="show" name="divcontent">
<dl class="content">
<dt>商戶訂單號 :</dt>
<dd>
<input id="WIDout_trade_no" name="WIDout_trade_no" />
</dd>
<hr class="one_line">
<dt>訂單名稱 :</dt>
<dd>
<input id="WIDsubject" name="WIDsubject" />
</dd>
<hr class="one_line">
<dt>付款金額 :</dt>
<dd>
<input id="WIDtotal_amount" name="WIDtotal_amount" />
</dd>
<hr class="one_line">
<dt>商品描述:</dt>
<dd>
<input id="WIDbody" name="WIDbody" />
</dd>
<hr class="one_line">
<dt></dt>
<dd id="btn-dd">
<span class="new-btn-login-sp">
<button class="new-btn-login" type="submit"
style="text-align: center;">付 款</button>
</span> <span class="note-help">若是您點擊「付款」按鈕,即表示您贊成該次的執行操做。</span>
</dd>
</dl>
</div>
</form>
</body>
</html>
alipaySuccess.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" %>
<html>
<head>
<title>支付成功!</title>
</head>
<body>
<h1 style="color: green;">購買成功</h1>
<table>
<tr>
<td>
訂單編號: ${out_trade_no }
</td>
</tr>
<td>
支付寶交易號: ${trade_no }
</td>
<tr>
</tr>
<td>
實付金額: ${total_amount }
</td>
<tr>
</tr>
<td>
購買產品:${productName }
</td>
</tr>
</table>
</body>
</html>
固然你能夠你要寫一個跳轉,到達支付頁面:
好了能夠吧:看下框圖吧:
好,感受已經將清楚了,注意複製jsp頁面時看清楚導入的包,不出意外,能夠成功的。