一、app公鑰和私鑰的生成
經過支付寶提供的工具便可下載並生成公鑰和私鑰,私鑰本身保留,公鑰要上傳到支付寶帳戶所在的應用便可。前端
二、接下來就是經過app支付api調後臺接口,後臺經過公鑰和私鑰等 一系列的參數生成一段鍵值對給前端APP:java
package com.qtay.gls.service.impl; import com.alipay.api.AlipayApiException; import com.alipay.api.AlipayClient; import com.alipay.api.DefaultAlipayClient; import com.alipay.api.domain.AlipayTradeAppPayModel; import com.alipay.api.request.AlipayTradeAppPayRequest; import com.alipay.api.response.AlipayTradeAppPayResponse; import com.qtay.gls.dao.entity.AlipayTradeModel; import com.qtay.gls.service.IPayService; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Service; @Service public class PayService implements IPayService { @Value("${alipay.app_id}") private String appId; @Value("${alipay.app_private_key}") private String appPrivateKey; @Value("${alipay.charset}") private String charset; @Value("${alipay.alipay_public_key}") private String alipayPublicKey; @Value("${alipay.sign_type}") private String signType; @Value("${alipay.notify_url}") private String notifyUrl; @Value("${alipay.server_url}") private String serverUrl; @Value("${alipay.format}") private String format; @Value("${alipay.product_code}") private String productCode; /** * 驗籤 * @param alipayTradeModel * @return * @throws AlipayApiException */ @Override public String getSign(AlipayTradeModel alipayTradeModel) throws AlipayApiException { //實例化客戶端 AlipayClient alipayClient = new DefaultAlipayClient( serverUrl, appId, appPrivateKey, format, charset, alipayPublicKey, signType); //實例化具體API對應的request類,類名稱和接口名稱對應,當前調用接口名稱:alipay.trade.app.pay AlipayTradeAppPayRequest request = new AlipayTradeAppPayRequest(); //SDK已經封裝掉了公共參數,這裏只須要傳入業務參數。如下方法爲sdk的model入參方式(model和biz_content同時存在的狀況下取biz_content)。 AlipayTradeAppPayModel model = new AlipayTradeAppPayModel(); model.setBody(alipayTradeModel.getBody()); model.setSubject(alipayTradeModel.getSubject()); model.setOutTradeNo(alipayTradeModel.getOutTradeNo()); model.setTimeoutExpress(alipayTradeModel.getTimeoutExpress()); model.setTotalAmount(alipayTradeModel.getTotalAmount()); model.setProductCode(productCode); request.setBizModel(model); request.setNotifyUrl(notifyUrl); //這裏和普通的接口調用不一樣,使用的是sdkExecute AlipayTradeAppPayResponse response = alipayClient.sdkExecute(request); System.out.println(response.getBody());//就是orderString 能夠直接給客戶端請求,無需再作處理。 return response.getBody(); } }
三、前端APP支付成功以後會調上面設置的notifyUrl(注意若是設置了權限要取消該url的jwt用戶認證,我這裏的url是:/alipay/notify)。調用的時候這裏有一個坑,也是我遇到的坑,在這裏跟你們分享一下:
1)個人這個controller存在亂碼問題,因此要設置web
@RequestMapping(value = "/alipay", produces = "application/json; charset=utf-8")
不然 會報亂碼錯誤,空指針什麼的spring
18-01-04 13:49:42.755 DEBUG [http-nio-9090-exec-1] o.s.web.servlet.DispatcherServlet - Could not complete request java.lang.NullPointerException: null
.antMatchers("/alipay/**").permitAll()
再一個就是,若是你頁面自己就出現亂碼問題,再經過下面這段代碼設置也是沒有用的,切記!json
//亂碼解決,這段代碼在出現亂碼時使用。 valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");
package com.qtay.gls.controller; import com.alipay.api.AlipayApiException; import com.alipay.api.internal.util.AlipaySignature; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Value; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; import javax.servlet.http.HttpServletRequest; import java.io.UnsupportedEncodingException; import java.util.HashMap; import java.util.Map; @RestController @RequestMapping(value = "/alipay", produces = "application/json; charset=utf-8") public class AlipayController { private Logger log = LoggerFactory.getLogger(AlipayController.class); @Value("${alipay.alipay_public_key}") private String alipayPublicKey; @Value("${alipay.charset}") private String charset; @Value("${alipay.sign_type}") private String signType; /** * 支付寶支付通知接口: * * @param * @return */ @PostMapping("/notify") public String notify(HttpServletRequest request) throws AlipayApiException, UnsupportedEncodingException { log.info("支付寶支付結果通知" + request.getParameterMap().toString()); //獲取支付寶POST過來反饋信息 Map<String, String> params = new HashMap<>(); Map requestParams = request.getParameterMap(); for (Object o : requestParams.keySet()) { String name = (String) o; 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(name + "=" + valueStr); } //切記alipaypublickey是支付寶的公鑰,請去open.alipay.com對應應用下查看。 //boolean AlipaySignature.rsaCheckV1(Map<String, String> params, String publicKey, String charset, String sign_type) boolean signVerified = AlipaySignature.rsaCheckV1(params, alipayPublicKey, charset, signType); //todo:flag爲何是false log.info("================== signVerified ==================" + signVerified); if (signVerified) { if ("TRADE_SUCCESS".equals(params.get("trade_status"))) { //付款金額 String amount = params.get("buyer_pay_amount"); //商戶訂單號 String out_trade_no = params.get("out_trade_no"); //支付寶交易號 String trade_no = params.get("trade_no"); log.info("amount=" + amount + ",out_trade_no=" + out_trade_no + ",trade_no=" + trade_no); return "success"; } } return "fail"; } }
參考文章:
一、一步一步帶你完成支付寶支付功能的集成(超詳細)
二、spring Boot 中文返回給瀏覽器亂碼 解析成問號?? fastJson jackJsonapi