Bootstrap

支付宝沙箱支付——傻瓜教程

个人网站想要接入支付,没有企业资质没办法接入。但是可以用沙箱支付测试一下。

1.进入支付宝开放平台首页,往下拉找到开发服务 > 沙箱
在这里插入图片描述

2.点击开放平台控制器,登录,进入研发服务
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
3.如上图所示,我们需要设置RSA2

作用:
RSA非对称
发起支付:商户(商户应用私钥签名) -> 支付宝(商户应用公钥验签)
异步通知:支付宝(支付宝私钥签名) -> 商户(支付宝公钥验签)

4.点击进入RSA应用私钥生成页面复制应用公钥。
回到上一张图那里,点开设置RSA2,粘贴应用公钥,生成支付宝公钥
在这里插入图片描述
5.IDEA新建项目pay,导入如下依赖

        <dependency>
            <groupId>com.alipay.sdk</groupId>
            <artifactId>alipay-sdk-java</artifactId>
            <version>4.16.2.ALL</version>
        </dependency>

新建controller,并创建类PayController,看下面注意事项

package com.jiafa.pay.controller;

import com.alipay.api.AlipayApiException;
import com.alipay.api.AlipayClient;
import com.alipay.api.DefaultAlipayClient;
import com.alipay.api.internal.util.AlipaySignature;
import com.alipay.api.request.AlipayTradePagePayRequest;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.*;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

/**
 * Created by jiafa
 * on 2021/8/6 23:20
 */
@Controller
@RequestMapping("/pay")
@Slf4j
public class PayController {
    private final String APP_ID = "";
    private final String APP_PRIVATE_KEY = "";
    private final String CHARSET = "UTF-8";
    private final String ALIPAY_PUBLIC_KEY = "";
    //这是沙箱接口路径,正式路径为https://openapi.alipay.com/gateway.do
    private final String GATEWAY_URL = "https://openapi.alipaydev.com/gateway.do";
    private final String FORMAT = "JSON";
    //签名方式
    private final String SIGN_TYPE = "RSA2";
    //支付宝异步通知路径,付款完毕后会异步调用本项目的方法,必须为公网地址
    private final String NOTIFY_URL = "http://fafa-xxxx.natapp1.cc/pay/returnUrl";
    //支付宝同步通知路径,也就是当付款完毕后跳转本项目的页面,可以不是公网地址
    private final String RETURN_URL = "http://127.0.0.1";

    @RequestMapping("alipay")
    public void alipay(Integer orderId, HttpServletResponse httpResponse) throws IOException {
        //实例化客户端,填入所需参数
        AlipayClient alipayClient = new DefaultAlipayClient(GATEWAY_URL, APP_ID, APP_PRIVATE_KEY, FORMAT, CHARSET, ALIPAY_PUBLIC_KEY, SIGN_TYPE);
        AlipayTradePagePayRequest request = new AlipayTradePagePayRequest();
        //在公共参数中设置回跳和通知地址
        request.setReturnUrl(RETURN_URL);
        request.setNotifyUrl(NOTIFY_URL);
        //根据订单编号,查询订单相关信息
//        Order order = IPayService.selectById(orderId);
        //商户订单号,商户网站订单系统中唯一订单号,必填
        String out_trade_no = orderId.toString();
        //付款金额,必填
        String total_amount = "0.01";
        //订单名称,必填
        String subject = "订单测试";
        //商品描述,可空
        String body = "";
        request.setBizContent("{\"out_trade_no\":\"" + out_trade_no + "\","
                + "\"total_amount\":\"" + total_amount + "\","
                + "\"subject\":\"" + subject + "\","
                + "\"body\":\"" + body + "\","
                + "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}");
        String form = "";
        try {
            form = alipayClient.pageExecute(request).getBody(); // 调用SDK生成表单
        } catch (AlipayApiException e) {
            e.printStackTrace();
        }
        httpResponse.setContentType("text/html;charset=" + CHARSET);
        httpResponse.getWriter().write(form);// 直接将完整的表单html输出到页面
        httpResponse.getWriter().flush();
        httpResponse.getWriter().close();
    }

    @RequestMapping(value = "/returnUrl", method = RequestMethod.GET)
    public String returnUrl(HttpServletRequest request, HttpServletResponse response)
            throws IOException, AlipayApiException {
        System.out.println("=================================同步回调=====================================");
        // 获取支付宝GET过来反馈信息
        Map<String, String> params = new HashMap<String, String>();
        Map<String, String[]> requestParams = request.getParameterMap();
        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("utf-8"), "utf-8");
            params.put(name, valueStr);
        }

        System.out.println(params);//查看参数都有哪些
        boolean signVerified = AlipaySignature.rsaCheckV1(params, ALIPAY_PUBLIC_KEY, CHARSET, SIGN_TYPE); // 调用SDK验证签名
//验证签名通过
        if(signVerified){
            // 商户订单号
            String out_trade_no = new String(request.getParameter("out_trade_no").getBytes("ISO-8859-1"), "UTF-8");

            // 支付宝交易号
            String trade_no = new String(request.getParameter("trade_no").getBytes("ISO-8859-1"), "UTF-8");

            // 付款金额
            String total_amount = new String(request.getParameter("total_amount").getBytes("ISO-8859-1"), "UTF-8");

            System.out.println("商户订单号="+out_trade_no);
            System.out.println("支付宝交易号="+trade_no);
            System.out.println("付款金额="+total_amount);

            //支付成功,修复支付状态
            // payService.updateById(Integer.valueOf(out_trade_no));
            return "ok";//跳转付款成功页面
        }else{
            return "no";//跳转付款失败页面
        }
    }
}

修改APP_IDAPP_PRIVATE_KEY(应用私钥)、ALIPAY_PUBLIC_KEY(支付宝公钥)、NOTIFY_URL(异步通知路径,必须是公网地址,如果没有公网地址怎么办?等等下面介绍一个内网穿透的办法。)、RETURN_URL,还有这里被我写死了,也可以通过查询orderId,获取Order对象,再写入信息。
在这里插入图片描述

这些信息改完就差不多了,我们来看一下内网穿透吧。

6.打开natapp网站,下载客户端。
需要购买隧道,咱们买个9元/月的就可以了
在这里插入图片描述
在这里插入图片描述
点开右边的配置,写上地址和端口号
在这里插入图片描述
7.打开刚刚下载的客户端,运行exe文件
在这里插入图片描述
在这里插入图片描述
8. 运行pay项目。访问以下路径:
http://fafa-xxxx.natapp1.cc/pay/alipay?orderId=112
显示支付界面
在这里插入图片描述
在刚刚配置RSA2那里有个沙箱账号,根据这个填入卖家信息即可。
在这里插入图片描述
9.下载沙箱钱包,用上图的卖家账号和密码登录,能看到支付记录。
在这里插入图片描述
10.完成,配置成功点个赞噢。

;