调用支付宝第三方接口(沙箱环境) SpringMVC+Maven
一.蚂蚁金服开放平台的操作
网址:https://open.alipay.com/platform/home.htm
支付宝扫码登陆
之后配置你的沙箱支付宝
支付宝提供一键生成工具便于开发者生成一对RSA2密钥:https://docs.open.alipay.com/291/105971
注意:生成时一定要选择PKCS8+2048,第一个坑
将应用网关和回调地址更改为:https://www.alipay.com
【AES密钥不用管】
【然后往下会有支付宝沙箱安卓端工具,下载,以供后续支付使用】
【沙箱里有沙箱账号,可以用其登陆,Don’t tell me you are 看不到,实在不行我提供给你,你发来邮箱,谢谢!!!!】
【还有要用买家账号登陆】
OK!到这基本配置就完了,可能有忘了的,后续再加吧,下面进入大家喜欢的代码时间:
公钥私钥工具下载
支付宝为技术开发人员提供一键生成工具,便于开发者生成一对 RSA 密钥、公钥证书申请CSR文件(在线申请应用公钥证书需要)。
应用公钥(public key)需提供给支付宝账号管理者上传到支付宝开放平台;应用私钥(private key)由开发者自己保存,需填写到代码中供签名时使用。加密的过程为系统使用公钥(public key)进行加密,并将密文发送到解密者,解密者用私钥(private key)解密将密文解码为明文。
开发者要保证接口中使用的私钥与此处的公钥匹配,否则无法调用接口。开发者可通过下方链接下载对应的密钥生成工具:
WINDOWS(windows版本工具请不要安装在含有空格的目录路径下)
MAC_OSX
WINDOWS老版本下载地址:WINDOWS
MAC OS老版本下载地址:MAC_OSX
注意:
密钥和应用(APPID)一一对应,即开发者需要为名下的每个应用分别设置密钥,且不同应用的密钥不能混用。
公钥私钥使用步骤
支付宝开放平台支持开发者使用普通公钥、公钥证书两种签名方式;下面分别向您介绍两种方式的工具操作步骤,包括如何使用密钥生成工具,生成应用公钥(public key)、应用私钥(private key)和公钥证书申请 CSR 文件;您也可以通过 视频版教程 快速了解。
普通公钥方式
1.下载相应环境工具并安装后即可使用,本步骤指引以 MAC_OSX 界面为例,如下图所示:
2.开发者根据开发语言选择密钥格式和密钥长度,新建应用请务必使用 2048 位(目前已使用 1024 位密钥长度的应用仍然可以正常调用接口,详情请见开放平台接口签名方式升级公告)。点击 生成密钥 后,工具会自动生成商户应用公钥(public key)和应用私钥(private key),如下图所示:
3.开发者点击工具界面下方的 打开文件位置,即可找到生成的公私钥文件,如下图所示:
4.生成密钥后,开发者就可以在应用的开发配置页面进行配置。点击 设置应用公钥 后,复制上一步生成的公钥,点击 保存,即可完成公钥的设置,如下图所示。
注意:
生成的私钥需妥善保管,避免遗失,不要泄露。应用私钥需填写到代码中供签名时使用。应用公钥需提供给支付宝账号管理者上传到支付宝开放平台。
二.配置一个用于支付宝调用的Config
package com.alipay.config;
import java.io.FileWriter;
import java.io.IOException;
/* *
*类名:AlipayConfig
*作者:AAA_有梦想一起实现
*/
public class AlipayConfig{
// ↓↓↓↓↓↓↓↓↓↓请在这里配置您的基本信息↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓
// 应用ID,您的APPID,收款账号既是您的APPID对应支付宝账号
public static String app_id = "APPID";//例:2016082600317257
// 商户私钥,您的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 = "你的服务器地址/项目名称/notify_url";
// 页面跳转同步通知页面路径 需http://格式的完整路径,不能加?id=123这类自定义参数,必须外网可以正常访问
/**
* 此页面是同步返回用户页面,也就是用户支付后看到的页面,上面的notify_url是异步返回商家操作,谢谢
* 要是看不懂就找度娘,或者多读几遍,或者去看支付宝第三方接口API,不看API直接拿去就用,遇坑不怪别人
*/
public static String return_url = " 你的服务器地址/项目名称/return_url";
// 签名方式
public static String sign_type = "RSA2";
// 字符编码格式
public static String charset = "gbk";
// 支付宝网关
public static String gatewayUrl = "https://openapi.alipaydev.com/gateway.do";
// 日志地址
public static String log_path = "D:/logs/";
// ↑↑↑↑↑↑↑↑↑↑请在这里配置您的基本信息↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑
/**
* 写日志,方便测试(看网站需求,也可以改成把记录存入数据库)
*
* @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();
}
}
}
}
}
如果你是在本地测试,支付完成不会跳转回调页面,那么就需要外网了,推荐一个东西,叫内网穿透,只要你电脑tomcat启动,可以连接外网,可以从127.0.0.1:8080或者其他端口也行进入你的项目,就可以用;
【内网穿透】
某花生qiao,不免费6块大洋---------------------------NATAPP 有免费的(只是域名会强制变),不过测试足够了----------------------ngrok或者frp自行度娘
也有免费开源的,如果不想麻烦就用【NATAPP】吧,网址貌似是:natapp.cn
三.控制器Controller里的代码
pom.xml里的依赖
如果不是maven项目,下载jar包导入
下载地址:http://central.maven.org/maven2/com/pentahohub/nexus/alipay-sdk-java/20150820220052/alipay-sdk-java-20150820220052.jar
Controller代码
/**
* 快捷支付调用支付宝支付接口
* @param model,id,payables,
* @throws IOException,AlipayApiException
* @return Object
* @author AAA_有梦想一起实现
* @date 2017年11月29日
*/
@RequestMapping("alipaySum")
public Object alipayIumpSum(Model model, String orderId, String body, HttpServletResponse response)
throws Exception {
// 获得初始化的AlipayClient
AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfigInfo.gatewayUrl, AlipayConfigInfo.app_id,
AlipayConfigInfo.merchant_private_key, "json", AlipayConfigInfo.charset,
AlipayConfigInfo.alipay_public_key, AlipayConfigInfo.sign_type);
// 设置请求参数
AlipayTradePagePayRequest alipayRequest = new AlipayTradePagePayRequest();
alipayRequest.setReturnUrl(AlipayConfigInfo.return_url);
alipayRequest.setNotifyUrl(AlipayConfigInfo.notify_url2);
//从数据库查出的数据
Order order = orderService.selectOrder(orderId);
// 付款金额,必填
alipayRequest.setBizContent("{\"out_trade_no\":\"" + order.getOrdCode() + "\"," + "\"total_amount\":\"" + order.getOrdMoney()
+ "\"," + "\"subject\":\"" + order.getOrdName() + "\"," + "\"body\":\"" + "明锐教育" + "\","
+ "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}");
// 请求
String result = alipayClient.pageExecute(alipayRequest).getBody();
// System.out.println(result);
AlipayConfigInfo.logResult(result);// 记录支付日志
response.setContentType("text/html; charset=gbk");
PrintWriter out = response.getWriter();
out.print(result);
return null;
}
参数传入是必须有的,不然会报错,说订单信息有误
如果有其他额外参数,请参考支付宝第三方API文档,这里菜鸟能力有限,懒得去查,也懒得写了
支付宝第三方支付API地址:https://docs.open.alipay.com/api_1/alipay.trade.create/
支付完成回调------异步返回商家,也就是notify_url
代码:
@RequestMapping("notify_url")
public void Notify(HttpServletResponse response, HttpServletRequest request) throws Exception {
System.out.println("----------------------------notify_url------------------------");
// 商户订单号
String out_trade_no = new String(request.getParameter("out_trade_no").getBytes("ISO-8859-1"), "GBK");
// 付款金额
String total_amount = new String(request.getParameter("total_amount").getBytes("ISO-8859-1"), "GBK");
// 支付宝交易号
String trade_no = new String(request.getParameter("trade_no").getBytes("ISO-8859-1"), "GBK");
// 交易说明
String cus = new String(request.getParameter("body").getBytes("ISO-8859-1"), "GBK");
// 交易状态
String trade_status = new String(request.getParameter("trade_status").getBytes("ISO-8859-1"), "GBK");
if (trade_status.equals("TRADE_SUCCESS")) {//支付成功商家操作
//下面是我写的一个简单的插入操作,根据你的操作自行编写
/*Map<Object, Object> map = new HashMap<Object, Object>();
map.put("cuId", Integer.valueOf(cus));
RepaymentPlan repaymentPlan = new RepaymentPlan();
Integer id = Integer.valueOf(out_trade_no);
double payablesCheck = Double.valueOf(total_amount);
RepaymentPlan repayCheck = serviceMain.selectByPrimaryKey(id);
double total = repayCheck.getPayables();
if (Double.valueOf(total_amount) < repayCheck.getPayables()) {
map.put("ubalance", total - Double.valueOf(total_amount));
serviceMain.updateCusMoney(map);
}
repaymentPlan.setId(id);
repaymentPlan.setActualPayment(total);
repaymentPlan.setRepaymentStatus(1);
int i = serviceMain.updateByPrimaryKeySelective(repaymentPlan);
System.out.println("---------------------还款影响行数----------------------------" + i);*/
}
}
关于代码中用到的Servlet的东西,请自行进行改进吧,还有不要像我一样把业务写在Controller里,要写在service里哦!
支付完成回调------同步返回用户的页面,也就是return_url,
代码:
/**
* 同步通知的页面的Controller
*
* @param request,response
* @throws InterruptedException
*/
@RequestMapping("return_url")
public String Return_url() throws InterruptedException {
return "alipayexit";
}
我这边就简单的返回了一个页面;
最后贴上请求支付接口的JSP的页面吧
<form name=alipayment action=alipay.trade.page.pay.jsp 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>
支付宝接口的SDK&DEMO地址:https://docs.open.alipay.com/270/106291/