全部的参数:
这里只介绍一下使用到的必填的参数:
公共请求参数:
业务请求参数:
响应参数(过程6的同步返回请求参数):
公共请求参数同上,业务请求参数有:
响应参数:
<dependency>
<groupId>commons-logginggroupId>
<artifactId>commons-loggingartifactId>
dependency>
<dependency>
<groupId>com.alipay.sdkgroupId>
<artifactId>alipay-sdk-javaartifactId>
dependency>
配置类用来指定公共的请求参数
要改的参数:
alipaydev
public class AlipayConfig {
//↓↓↓↓↓↓↓↓↓↓请在这里配置您的基本信息↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓
// 应用ID,您的APPID,收款账号既是您的APPID对应支付宝账号
public static String app_id = "111111111";
// 商户私钥,您的PKCS8格式RSA2私钥
public static String merchant_private_key = "私钥";
// 支付宝公钥,查看地址:https://openhome.alipay.com/platform/keyManage.htm 对应APPID下的支付宝公钥。
public static String alipay_public_key = "公钥";
// 服务器异步通知页面路径 需http://格式的完整路径,不能加?id=123这类自定义参数,必须外网可以正常访问
public static String notify_url = "http://localhost:9090/notify_url.jsp";
// 页面跳转同步通知页面路径 需http://格式的完整路径,不能加?id=123这类自定义参数,必须外网可以正常访问
public static String return_url = "http://localhost:9090/p2p/loan/back";
// 签名方式
public static String sign_type = "RSA2";
// 字符编码格式
public static String charset = "utf-8";
// 支付宝网关
public static String gatewayUrl = "https://openapi.alipaydev.com/gateway.do";
// 支付宝网关
public static String log_path = "C:\\";
//↑↑↑↑↑↑↑↑↑↑请在这里配置您的基本信息↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑
/**
* 写日志,方便测试(看网站需求,也可以改成把记录存入数据库)
* @param sWord 要写入日志里的文本内容
*/
public static void logResult(String sWord) {
FileWriter writer = null;
try {
writer = new FileWriter(log_path + "alipay_log_" + System.currentTimeMillis()+".txt");
writer.write(sWord);
} catch (Exception e) {
e.printStackTrace();
} finally {
if (writer != null) {
try {
writer.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
由于支付这个功能是公共的,所以把这个功能单独提出来做为一个pay模块,而页面是在web模块,从web访问pay的两种方式:
后台接收到充值请求后,访问pay工程的Alipay方法,Alipay方法中访问了支付宝的统一收单下单并支付页面接口
/**
* @param rechargeMoney:充值金额
* @return:跳转到一个页面
*/
@RequestMapping("/loan/toAlipayRecharge")
public String toAlipayRecharge(HttpServletRequest request,
Double rechargeMoney,
Model model) {
model.addAttribute("out_trade_no",rechargeNo);
model.addAttribute("total_amount", rechargeMoney);
model.addAttribute("subject", rechargeRecord.getRechargeDesc());
return "p2pToPay";
}
p2pToPay.html:在页面加载完成后,自动调用pay模块的方法
<form method="post" action="http://localhost:9094/pay/api/alipay">
<input type="hidden" name="out_trade_no" th:value="${out_trade_no}">
<input type="hidden" name="total_amount" th:value="${total_amount}">
<input type="hidden" name="subject" th:value="${subject}">
form>
<script>document.forms[0].submit()script>
pay模块的alipay方法,该模块的负责向支付宝统一收单下单并支付页面接口发起请求,获取到一个类似上面 p2pToPay.html 页面的字符串,将该字符串放到一个页面中,即可跳转到支付宝的付款页面
@RequestMapping("/api/alipay")
public String alipay(Model model,
String out_trade_no,
String total_amount,
String subject) throws AlipayApiException {
//获得初始化的AlipayClient,从AlipayConfig中获取公共请求参数
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); 有需要的设,这里就不设置了
//拼接请求的参数
alipayRequest.setBizContent("{\"out_trade_no\":\""+ out_trade_no +"\","
+ "\"total_amount\":\""+ total_amount +"\","
+ "\"subject\":\""+ subject +"\","
+ "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}");
//请求
String result = alipayClient.pageExecute(alipayRequest).getBody();
//输出
model.addAttribute("result", result);
return "/payToAlipay";
}
这里的使用的是jsp页面,payToAlipay.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Titletitle>
head>
<body>
${result}
body>
html>
@RequestMapping("/loan/back")
public String alipayBack(HttpServletRequest request,
Model model,
String out_trade_no) throws Exception {
//转换后的请求参数
Map<String,String> params = new HashMap<String,String>();
//请求参数
Map<String,String[]> requestParams = request.getParameterMap();
//这段代码将请求参数的String[]转为使用逗号隔开的String字符串
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验证签名
if(signVerified){
Map<String ,Object > map = new HashMap<>();
map.put("out_trade_no", out_trade_no);
/*
在这里调用通过httpClint调用支付宝统一收单线下交易查询接口,查询交易是否成功
*/
// 创建httpClient对象
CloseableHttpClient httpClient = HttpClients.createDefault();
// 创建http对象
HttpPost httpPost = new HttpPost(url);
/**
* setConnectTimeout:设置连接超时时间,单位毫秒。
* setConnectionRequestTimeout:设置从connect Manager(连接池)获取Connection
* 超时时间,单位毫秒。这个属性是新加的属性,因为目前版本是可以共享连接池的。
* setSocketTimeout:请求获取数据的超时时间(即响应时间),单位毫秒。 如果访问一个接口,多少时间内无法返回数据,就直接放弃此次调用。
*/
RequestConfig requestConfig = RequestConfig.custom().setConnectTimeout(CONNECT_TIMEOUT).setSocketTimeout(SOCKET_TIMEOUT).build();
httpPost.setConfig(requestConfig);
// 封装请求参数
packageParam(params, map);
// 创建httpResponse对象
CloseableHttpResponse httpResponse = null;
String result = "";
try {
// 执行请求
httpResponse = httpClient.execute(httpPost);
// 获取返回结果
if (httpResponse != null && httpResponse.getStatusLine() != null) {
if (httpResponse.getEntity() != null) {
result = EntityUtils.toString(httpResponse.getEntity(), ENCODING);
}
}
} finally {
// 释放资源
release(httpResponse, httpClient);
}
//result为响应回来的json字符串,根据查询结果做出相应的业务处理
}
官方文档:使用沙箱环境
使用沙箱环境可以在支付宝开发环境下完成一些主要功能和主要逻辑,比如使用沙箱环境进行支付,收款等