一、对接前的准备
本文对接使用的是支付宝的电脑网站支付,其它对接方式可做参考
接入分两种
第一种是常规接入,主要是针对企业用户,需要有营业执照;经过备案能正常访问的网站
第二种是沙箱接入,主要针对个人用户,只需要有能正常使用的支付宝就可以完成接入,真实企业用户在开发过程种也可使用沙箱环境与线上数据隔离。
本文使用的是沙箱接入,也会涉及常规接入的使用
1.1创建应用,获取参数:APPID(使用沙箱环境可跳过)
打开: 支付宝开放平台,选择网页/移动应用开发
==注意:==阿里的开放平台会有更新,排版可能会有差异,本文发帖时间为2022-08-01
选择开发文档
点击开发
可以看到开发前的流程,如果是使用企业用户,还需要一个签约功能才能正常使用
点击创建应用,登录开放平台控制台
扫码登录
点击创建网页/移动应用
输入必填项后,确认创建
这里可以获取第一个需要的参数 :APPID,复制到文本种,保存
1.1.1添加产品
添加完成后是为开通,需要后续签约应用
1.1.2配置密钥,获取第二个参数:商户的私钥
本文使用的是密钥方式,如需使用企业对个人转账,如退款需使用证书模式
点击下载密钥生成器
安装完成后进入
点击生成密钥,然后复制公钥到支付宝页面
保存设置
第二个参数:商户的私钥需要复制出来 自己保留
1.1.3支付宝网关
复制出来即可
1.1.4生成参数密钥对
复制出来即可
1.2绑定应用,获取参数:PID(使用沙箱环境可跳过)
进入商家服务平台
复制收单账号
1.3上线应用(使用沙箱环境可跳过)
在 开放平台,控制台中找到应用,上线点击提交审核即可
1.4签约(使用沙箱环境可跳过)
需上传企业营业执照等信息
前往商家中心—产品中心—电脑网站师傅
需要提交如下资料,
2 获取沙箱配置(企业用户可跳过)
进入 沙箱环境,点击沙箱账号
复制出需要的参数即可
安卓手机可以下载支付宝沙箱版测试用
二、JAVA接入支付宝支付
2.1统一下单接单(新版本的返回下单页面)
引入Alipay SDK
<dependency>
<groupId>com.alipay.sdk</groupId>
<artifactId>alipay-sdk-java</artifactId>
<version>4.31.72.ALL</version>
</dependency>
添加配置文件 alipay-sandbox.properties
# 支付宝支付相关参数
# 应用ID ,APPID
alipay.app.id=2021000
# 商户PID
alipay.seller-id=208862198
# 支付宝网关
alipay.gateway-url=https://openapi.alipaydev.com/gateway.do
#商户私钥 RSA2私钥
alipay.merchant-private-key=/IE7GuL7ncnxUFofSBQpS7vdbvVNqmuI+1aT8ratWN9QvPdjjrKXmo79iR+KHe+gz2NsC8nnjsV5sTA21mbnnnO3IUg1qJkbpL7lgKSE3GjH/f+8IGgllIKac/2UfWxbjaVjuSCxND5yHgpZLagO6DxkKID68czt/r602avcJtJ1x6TMwrYyvVrtuSBvy/9iNJPKN8aIt596znIAX+8XXJx7UJ9B1NwoaAUbwvu0Wdk4v9Vw55IOZGIUHVwfW3K3ol2QWPlxIb8cxK4pwvcnk0+dcZ2cU7YDNO0DYYoy4sIXLBvn52TR/FPAgMBAAECggEAQ8itCkCVOKb4O3g3WGCcwXAbQSjMcTuJAZGhEd5auKc0n4ZNYFVKmUg2tDhdzHoQbpV+sDUBZS9+5RVX538+AcKHlZaDCsmzEIHvG86MtYVTry69zZtzfEMASdvwH+VmJZD16NR8ctLJxPk4+iTYm1v1jONojizF4MuEV37NYs1CXcDPzg3iW3QYuTCgqzyD6U1XB8BwpzzxA0spvU97TuTtsuqom57fZVTwvqeyIUcR9t86m4Yt4oSH5k1cWc+rxvcsa+tIoXFo8x4NO/RB0H5Pnpxr5RkLz+PfJ2P4TnN2XunZHFI5GfuxWS6Dry81aV2yKeFp/sDUDED65h8wYQKBgQDPv/sZihsfRhFaCwAr/NWTgv12/rPbPULbsCOVO2mS7KDn4gM4b9xqUVtCmwjVAPVSK4K3RipcmKA4rNa0KH+0Ku109L5ydlErcHTnLJwFirwGezKHwCzT+yBYATLKg4NRJmDDcMJtTQA2tmA3YIC7HKFfjUqnWUR6z6bhucl/GQKBgQCg0q7HtX/a0FjB2RyCSnH03Ujeq250FuLPC/F5EiRx5YkGDnSvvaGdCwXHEYRwN75FIMAssi5hs1EU3OQVYgEtnGDwKvKgmdZySL4g+TfZezmMeoyokZV33KI/3lk16x9IYAR9LCGCMyghWbEd+4EM/WJJavdQ6vShC/sGy69IpwKBgHkhcwMdJ1tAu3VI3LzJGq57vdXYBH5cZdM6DEVC8vebyOXrPf12G4pSDWf0hV3MxT036Wt7GdALnb/t5vH8exlNvk5nNXP/0KwHUeJIfGAu2BrfUkMcpgajceReLoMt0y9JtTm/UV3xe6JrDAa92dE/jEfuVgzlW6xPzvnmHbphAoGAWHCDvT+KeAJrTO7gRqY51LZ8BDeyHhUX1VR0Dmhzsk7P84yvjpVx8rLFEpwHgM6my80e4XV7HC9IP3jZ1Qh4LWT5yhlUJA11aJOoOunSVL72/tHF2E13LNsgPo8/7+7E3UAwN8W1B6yqPOzeAeb1KPeOvWEdcFpE/OthuHL6ibECgYEArFSKaurnzbs01mFICz5P5kDoytM3cII9oZsDktdUVtB6roaVc4gbP4BSCwUH3Abkkm8mNJdcXZB0gRmn+JGhJ0m6CvGCLEefmMndxHuYQAtDYJO78XvAPHokx80U0RSWRQZ6/fEygHctmu9Bj/kHYXiH1lxIo1Dbcu4rvJb9LXc=
# 支付宝公钥
alipay.alipay-public-key=/+BlPxc8HG/4p2r+Z4EIzU7gFKkuE1I2xnDdJ+Dm1l//vYsArvtNwIKpOr0/oNfsNLhxhrswX7WEW3tyQopEU//KAhTXpsT0KTeKo8dxpl4FOJ9jrnkdVnxg/I2xY/oM7LeGoREiecJulcCa5cZbEct1OdnLzeQVomLlXUhBXZWyx1OVyzaQDk+X+yA2VY90uPvFcqFRBR5k3Y3fN7/39CTQvYWl+wvMcY4TFwO4j7hvLim9vgI3iyn2rOObMkGTJPlL9VNU5hBnRflBSjWAwIDAQAB
# 接口内容加密密钥 对此密钥
alipay.content-key=+ZyQ==
#页面跳转同步通知页面 测试地址 后期需要修改为实际地址
alipay.return-url=https://www.baidu.com/
修改为自己的支付参数即可
修改配置文件为springBoot的配置,绿色的小叶子
点加号找到刚才添加的配置文件
封装签名过程
import com.alipay.api.*;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;
import org.springframework.core.env.Environment;
import javax.annotation.Resource;
/**
* @author:xianyu
* @createDate:2022/7/30
* @description:
*/
@Configuration
@PropertySource("classpath:alipay-sandbox.properties")
public class AlipayClientConfig {
@Resource
private Environment config;
@Bean
public AlipayClient alipayClient() throws AlipayApiException {
AlipayConfig alipayConfig = new AlipayConfig();
//设置网关地址
alipayConfig.setServerUrl(config.getProperty("alipay.gateway-url"));
//设置应用ID
alipayConfig.setAppId(config.getProperty("alipay.app.id"));
//设置应用私钥
alipayConfig.setPrivateKey(config.getProperty("alipay.merchant-private-key"));
//设置请求格式,固定值json
alipayConfig.setFormat(AlipayConstants.FORMAT_JSON);
//设置字符集
alipayConfig.setCharset(AlipayConstants.CHARSET_UTF8);
//设置支付宝公钥
alipayConfig.setAlipayPublicKey(config.getProperty("alipay.alipay-public-key"));
//设置签名类型
alipayConfig.setSignType(AlipayConstants.SIGN_TYPE_RSA2);
//构造client
AlipayClient alipayClient = new DefaultAlipayClient(alipayConfig);
return alipayClient;
}
}
编写demo
controller层
import com.mcsgis.saas.yun.service.AlipayNewService;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
/**
* @author:xianyu
* @createDate:2022/7/30
* @description:
*/
@CrossOrigin
@RestController
@RequestMapping("/yun")
@Api(tags = "网站支付宝支付")
@Slf4j
public class AlipayController {
@Autowired
private AlipayNewService alipayService;
@ApiOperation("统一收单下单并支付接口调用")
@GetMapping("/scanPay")
public String tradePagePay() throws Exception {
log.info("统一收单下单并支付接口调用");
/*
支付宝开放平台接收 request 请求对象后
会为开发者生成一个html形式的form表单,包含自动提交的脚本
*/
String formStr = alipayService.tradeCreate();
/*
将form表单字符串返回给前端
前端自动提交脚本
表单会自动提交到 action熟悉只想的支付宝开放平台中 为用户展示一个支付页面
*/
return formStr;
}
}
service
/**
* @author:xianyu
* @createDate:2022/8/1
* @description:
*/
public interface AlipayNewService {
/**
* 支付宝开放平台接收 request请求对象后
* @return
*/
String tradeCreate() throws Exception;
}
serviceImpl
import com.alipay.api.AlipayClient;
import com.alipay.api.domain.AlipayTradePagePayModel;
import com.alipay.api.request.AlipayTradePagePayRequest;
import com.alipay.api.response.AlipayTradePagePayResponse;
import com.mcsgis.saas.yun.service.AlipayNewService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.configurationprocessor.json.JSONObject;
import org.springframework.core.env.Environment;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
/**
* @author:xianyu
* @createDate:2022/8/1
* @description:
*/
@Slf4j
@Service
public class AlipayNewServiceImpl implements AlipayNewService {
@Autowired
private AlipayClient alipayClient;
@Autowired
private Environment config;
@Override
@Transactional
public String tradeCreate() throws Exception {
AlipayTradePagePayRequest request = new AlipayTradePagePayRequest();
//支付宝公共参数
//request.setNotifyUrl("");
request.setReturnUrl(config.getProperty("alipay.return-url"));
//原始方式封装业务参数
JSONObject bizContent = new JSONObject();
bizContent.put("out_trade_no", "20220801113100005");
bizContent.put("total_amount",100 );
bizContent.put("subject", "玛莎拉蒂");
bizContent.put("product_code", "FAST_INSTANT_TRADE_PAY");
//面向对象封装业务参数
AlipayTradePagePayModel model =new AlipayTradePagePayModel();
model.setOutTradeNo("20220801113100005");
model.setTotalAmount("100");
model.setSubject("玛莎拉蒂");
model.setProductCode("FAST_INSTANT_TRADE_PAY");
//bizContent.put("time_expire", "2022-08-01 22:00:00");
商品明细信息,按需传入
//JSONArray goodsDetail = new JSONArray();
//JSONObject goods1 = new JSONObject();
//goods1.put("goods_id", "goodsNo1");
//goods1.put("goods_name", "子商品1");
//goods1.put("quantity", 1);
//goods1.put("price", 0.01);
//goodsDetail.add(goods1);
//bizContent.put("goods_detail", goodsDetail);
扩展信息,按需传入
//JSONObject extendParams = new JSONObject();
//extendParams.put("sys_service_provider_id", "2088511833207846");
//bizContent.put("extend_params", extendParams);
//request.setBizContent(bizContent.toString());
request.setBizModel(model);
//执行请求,调用支付宝
AlipayTradePagePayResponse response = alipayClient.pageExecute(request);
if (response.isSuccess()) {
log.info("调用成功,返回结果:[{}]",response.getBody());
return response.getBody();
} else {
log.info("调用失败!!");
}
return null;
}
}
这里的返回值是一个自动提交的form表单,测试时不能使用postman,需要用浏览器测试
浏览器输入url地址
自动跳转到扫码支付
2.2支付回调
配置文件里面加上 alipay.notify-url ,我这是内网开发,用到了ngrok内网穿透
ngrok官网
下载之后,解压
点击运行
看官网的操作流程 80是指定的端口,可以按需修改
ngrok config add-authtoken 2CkM5EH6uj3nZwfIWdyxNXZ3Pb2_86cjLwcfPir1qkiGNRjhv
ngrok http 80
alipay.notify-url=https://f12d-171-113-169-92.jp.ngrok.io/yun/tradeNotify
这个是国外的,官网可能有点慢,国内的有natapp,不过ip会变,最方便的是花生壳现在穿透要6块钱认证,这里就不贴出了
回调controller
/**
* 支付宝回调
*
* @param params
* @return
*/
@ApiOperation("交易通知")
@PostMapping("/tradeNotify")
public String tradeNotify(@RequestParam Map<String, String> params) {
log.info("支付通知,正在执行,通知参数:{}", JSON.toJSONString(params));
return alipayService.tradeNotify(params);
}
service
@Override
public String tradeNotify(Map<String, String> params) {
String result = "failure";
try {
//异步通知验签
boolean signVerified = AlipaySignature.rsaCheckV1(params,
config.getProperty("alipay.alipay-public-key"),
AlipayConstants.CHARSET_UTF8,
AlipayConstants.SIGN_TYPE_RSA2);
if (!signVerified) {
// TODO 验签失败则记录异常日志,并在response中返回failure.
log.error("支付成功,异步通知验签失败!");
return result;
}
log.info("支付成功,异步通知验签成功!");
//TODO 验签成功后,按照支付结果异步通知中的描述,对支付结果中的业务内容进行二次校验
//1.验证out_trade_no 是否为商家系统中创建的订单号
String outTradeNo = params.get("out_trade_no");
//2.判断 total_amount 是否确实为该订单的实际金额
String totalAmount = params.get("total_amount");
//3.校验通知中的 seller_id是否为 out_trade_no 这笔单据的对应的操作方
String sellerId = params.get("seller_id");
if (!sellerId.equals(config.getProperty("alipay.seller-id"))) {
log.error("商家PID校验失败");
return result;
}
//4.验证 app_id 是否为该商家本身
String appId = params.get("app_id");
if (!appId.equals(config.getProperty("alipay.app.id"))){
log.error("app_id校验失败");
return result;
}
//在支付宝的业务通知中,只有交易通知状态为 TRADE_SUCCESS 或 TRADE_FINISHED 时,支付宝才会认定为买家付款成功
String tradeStatus = params.get("trade_status");
if (!"TRADE_SUCCESS".equals(tradeStatus) && !"TRADE_FINISHED".equals(tradeStatus)){
log.error("支付未成功");
return result;
}
//TODO 处理自身业务
result = "success";
} catch (AlipayApiException e) {
e.printStackTrace();
}
return result;
}
到这里基本的支付业务需求就能满足了文章来源:https://www.toymoban.com/news/detail-432210.html
====================待续文章来源地址https://www.toymoban.com/news/detail-432210.html
到了这里,关于【超详细,全流程】java对接支付宝支付的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!