因为公司项目需要集成支付宝支付,所以这次我把我集成支付宝支付的过程记录一下
1接入准备
接入准备:https://opensupport.alipay.com/support/helpcenter/190/201602469553?ant_source=antsupport
支付宝的开发文档里的接入准备写的很详细,根据开发文档一步步来就可以了
2服务端demo
我先放一下主要系统交互图
我的理解服务端要做的事就是
1、准备好支付宝需要的各种参数传入到支付宝给的接口中,然后支付宝返回给你支付请求。然后IOS或者安卓获取这串支付请求去调起支付宝的收银界面,如果可以正常调起,那说明这一步没问题。
2、支付完成后支付宝会去调用你设置的异步回调接口,你需要在回调接口中判断用户是否支付完成
首先我们先引入支付宝的依赖
<!--添加支付宝的依赖-->
<dependency>
<groupId>com.alipay.sdk</groupId>
<artifactId>alipay-sdk-java</artifactId>
<version>4.8.10.ALL</version>
</dependency>
服务端官网的开发文档:https://opendocs.alipay.com/open/54/106370/
private final static String APP_ID = "你的支付宝APPID";
private final static String APP_PRIVATE_KEY = "你的应用私钥";
private final static String CHARSET = "UTF-8";
private final static String ALIPAY_PUBLIC_KEY = "";
private final static String URL = "https://openapi.alipay.com/gateway.do";
private static String CER_PATH = "/opt/aliPay/"; //服务器支付宝证书
// private static String CER_PATH="E:\\ALiPayCert\\"; //本地Windows测试路径
/**
* JAVA服务端 SDK 生成 APP支付订单息
*/
public static String generatePayOrderInformation() throws AlipayApiException {
//构造client
CertAlipayRequest certAlipayRequest = new CertAlipayRequest();
//设置网关地址
certAlipayRequest.setServerUrl(URL);
//设置应用Id
certAlipayRequest.setAppId(APP_ID);
//设置应用私钥
certAlipayRequest.setPrivateKey(APP_PRIVATE_KEY );
//设置请求格式,固定值json
certAlipayRequest.setFormat("json");
//设置字符集
certAlipayRequest.setCharset(CHARSET);
//设置签名类型
certAlipayRequest.setSignType("RSA2");
//设置应用公钥证书路径
certAlipayRequest.setCertPath(CER_PATH+"你的应用公钥证书路径");
//设置支付宝公钥证书路径
certAlipayRequest.setAlipayPublicCertPath(CER_PATH+"你的支付宝公钥证书路径");
//设置支付宝根证书路径
certAlipayRequest.setRootCertPath(CER_PATH+"你的支付宝根证书路径");
//构造client
AlipayClient alipayClient = new DefaultAlipayClient(certAlipayRequest);
//实例化具体API对应的request类,类名称和接口名称对应,当前调用接口名称:alipay.trade.app.pay
AlipayTradeAppPayRequest request = new AlipayTradeAppPayRequest();
//SDK已经封装掉了公共参数,这里只需要传入业务参数。以下方法为sdk的model入参方式(model和biz_content同时存在的情况下取biz_content)。
AlipayTradeAppPayModel model = new AlipayTradeAppPayModel();
model.setBody("这是一双鞋子"); //对一笔交易的具体描述信息
model.setSubject("鞋子"); //商品的标题/交易标题/订单标题/订单关键字等
model.setOutTradeNo("1231231"); //商户网站唯一订单号
model.setTimeoutExpress(30m); //该笔订单允许的最晚付款时间,逾期将关闭交易。取值范围:1m~15d。m-分钟,h-小时,d-天,1c-当天(1c-当天的情况下,无论交易何时创建,都在0点关闭)。该参数数值不接受小数点,如 1.5h,可转换为 90m。
model.setTotalAmount(100); //订单总金额,单位为元,精确到小数点后两位
model.setProductCode("QUICK_MSECURITY_PAY"); //销售产品码,商家和支付宝签约的产品码,APP支付功能中该值固定为:QUICK_MSECURITY_PAY
request.setBizModel(model);
request.setNotifyUrl("http://你的回调路径/aliPaySuccess");//商户外网可以访问的异步地址
try {
//这里和普通的接口调用不同,使用的是sdkExecute
AlipayTradeAppPayResponse response = alipayClient.sdkExecute(request);
System.out.println(response.getBody());//就是orderString 可以直接给客户端请求,无需再做处理。
return response.getBody();
} catch (AlipayApiException e) {
e.printStackTrace();
return null;
}
}
我这里接口加签方式选择的是公钥证书如果你选择的是普通公钥方式应该只需要把公钥跟私钥传入代码即可。上面获取的response.getBody()就是传给客户端的支付请求,如果客户端可以调出支付宝的收银台说明代码没问题。我们也可以使用支付宝给的调试工具进行调试
调试工具:https://openclub.alipay.com/club/history/read/7695
截下来我们就要编写上面我们设置的异步回调接口的代码
/***
* 支付宝成功回调
* @param request
* @return
* @throws Exception
*/
@RequestMapping("aliPaySuccess")
public Object aliPaySuccess(HttpServletRequest request) {
Map<String, Object> map = new HashMap<>();
try {
Map<String, String> params = new HashMap<>();
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] + ",";
}
// 乱码解决,这段代码在出现乱码时使用。
// valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");
params.put(name, valueStr);
}
String orderNumber = request.getParameter("out_trade_no"); //获取订单号
String total_amount = request.getParameter("total_amount"); //获取总金额
String trade_status = request.getParameter("trade_status"); //交易状态
if(trade_status.equals("TRADE_FINISHED") || trade_status.equals("TRADE_SUCCESS")){
//判断该笔订单是否在商户网站中已经做过处理(可参考“集成教程”中“3.4返回数据处理”)
//如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序
//如果有做过处理,不执行商户的业务程序
System.out.println("================================交易状态为:"+trade_status+"====================================");
System.out.println("================================订单编号为:"+orderNumber+"====================================");
System.out.println("================================订单金额为:"+total_amount+"====================================");
//支付成功后进行修改订单等操作
return "SUCCESS";
} else {
map.put("returnValue", 2);
map.put("errMsg", "订单未支付完成");
System.out.println("================================订单未支付完成====================================");
return map;
}
}catch (Exception e){
e.printStackTrace();
map.put("returnValue", 99);
map.put("errMsg", "系统错误,请不要违法。");
System.out.println("================================系统错误====================================");
return map;
}
}
官网文档:https://opendocs.alipay.com/open/204/105301/#s1
测试结果在控制台打印出来,金额,订单号跟我们创建的订单一样说明服务端集成支付宝支付就算成功了!