25-javaweb接入支付寶支付接口

 

 

想熟悉支付寶接口支付,後面可能會用,不如在課設中試試手。好吧據說支付寶不微信支付要簡單些,就拿支付寶的先練下手吧。html

基本學習流程,百度一下,找篇博客看下。java

推薦下面這個篇博客,講的挺好的,複製過來。git

固然在最後我會把我遇到的問題寫出來,避免後人在走彎路,並將支付寶的demo整合到個人ssm項目中,同時,會詳細講下如何將demo中的jsp跟據需求改成contral層。github

比較能夠的操做流程:別人寫的很好了,我就不想寫了:https://github.com/OUYANGSIHAI/sihai-maven-ssm-alipayexpress

1、支付寶測試環境代碼測試

1.下載電腦網站的官方demo:

下載地址:https://docs.open.alipay.com/270/106291/json

image

2.下載解壓導入eclipse

image

readme.txt請好好看一下。服務器

只有一個Java配置類,其他都是JSP。微信

3.配置AlipayConfig

(1).註冊螞蟻金服開發者帳號(免費,不像蘋果會收取費用)app

註冊地址:https://open.alipay.com ,用你的支付寶帳號掃碼登陸,完善我的信息,選擇服務類型(我選的是自研)。框架

image

(2).設置app_id和gatewayUrl

image

image

其中密鑰須要本身生成,appID和支付寶網關是已經給好的,網關有dev字樣,代表是用於開發測試。

(3).設置密鑰

image

點擊「生成方法」,打開界面以下:

image

下週密鑰生成工具,解壓打開後,選擇2048位生成密鑰:

image

若是沒有設置過,此時顯示文本是「設置應用公鑰」,我這裏是已經設置過得。

image

設置方法,「打開密鑰文件路徑」:

image

image

複製應用公鑰2048.txt中的內容到點擊「設置應用公鑰」的彈出框中,保存:

image

  • 商戶私鑰(merchant_private_key)

    複製 應用私鑰2048.txt 中的內容到merchant_private_key中。

  • 支付寶公鑰(alipay_public_key)

image

點擊如上圖連接,複製彈出框裏面的內容到alipay_public_key。

若是這個設置不對,結果是:支付成功,可是驗籤失敗。

若是是正式環境,須要上傳到對應的應用中:

image

(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

4.測試運行

image

image

測試用的支付寶買家帳戶能夠在「沙箱帳號」這個頁面能夠找到:

image

支付成功後,驗簽結果:

image

 

 

好吧,要是你成功跑起來的那麼你很幸運,沒有到什麼坑,我遇到了以下坑點。記錄一下,以供參考:

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頁面時看清楚導入的包,不出意外,能夠成功的。

相關文章
相關標籤/搜索