支付宝调用支付流程(沙箱环境)

news2025/1/11 17:09:18

文章目录

    • 实现效果:
    • 前提准备
    • 支付流程
      • 方案一
        • 1. 导入依赖
        • 2. 配置文件
        • 3. 支付宝初始化
        • 4. 唤起支付
      • 方案二
        • 1. 导入依赖
        • 2. 唤起支付

实现效果:

在这里插入图片描述

前提准备

由于本文只是提及支付的流程及其一些相关知识点,所以前提数据自行准备,参考支付宝支付接口的调用

支付流程

举例说明:
如果平台只是给某商家使用,则全平台全部的收款都是一个,但是平台存在多个商家,各自收款,每个商家都存在不同的账户,则会出现多个收款方。
由于平台不确定有一个商家,即有几个收款方,所以,提供了两种方案唤起支付宝支付,
如果平台只有一个收款方,则使用方案一:单收款方;
如果平台存在多个收款方,则使用方案二:多收款方;

方案一

1. 导入依赖

<!--aliPay-->
<dependency>
    <groupId>com.alipay.sdk</groupId>
    <artifactId>alipay-easysdk</artifactId>
    <version>2.2.0</version>
</dependency>

2. 配置文件

alipay.appId=2021000116671219
# 商户私钥
alipay.privateKey=参数自备
# 支付宝公钥
alipay.publicKey=参数自备
# 服务器异步通知页面路径 ,需要公网能访问到。
alipay.notifyUrl=回调地址,需要根据自己实际的回调地址填写
# 页面跳转同步通知页面路径 需要公网能访问到。支付完成之后页面跳转url
alipay.returnUrl=https://www.baidu.com
# 签名方式
alipay.signType=RSA2
# 字符编码格式
alipay.charset=utf-8
# 支付宝网关
alipay.gateway=openapi.alipaydev.com
# 支付宝日志
alipay.logPath="C:\\"

3. 支付宝初始化

创建一个类,实现ApplicationRunner,这里面的值都是从配置文件获取。
如图
代码:

package com.mallshop.init;

import com.alipay.easysdk.factory.Factory;
import com.alipay.easysdk.kernel.Config;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.stereotype.Component;

/**
 * 项目初始化
 *
 * @author
 * @date 2020-09-08
 */
@Component
public class ProjectInit implements ApplicationRunner {

    //应用id
    @Value("${alipay.appId}")
    private String appId;

    //私钥
    @Value("${alipay.privateKey}")
    private String privateKey;

    //公钥
    @Value("${alipay.publicKey}")
    private String publicKey;

    //支付宝网关
    @Value("${alipay.gateway}")
    private String gateway;

    //支付成功后的接口回调地址,不是回调的友好页面,不要弄混了
    @Value("${alipay.notifyUrl}")
    private String notifyUrl;

    /**
     * 项目初始化事件
     *
     */
    @Override
    public void run(ApplicationArguments args) throws Exception {
        //初始化支付宝SDK,使用Factory,设置客户端参数,只需设置一次,即可反复使用各种场景下的API Client
        Factory.setOptions(getOptions());
        System.out.println("**********支付宝SDK初始化完成**********");
    }

    private Config getOptions() {
        //这里省略了一些不必要的配置,可参考文档的说明

        Config config = new Config();
        config.protocol = "https";
        config.gatewayHost = this.gateway;
        config.signType = "RSA2";

        config.appId = this.appId;

        // 为避免私钥随源码泄露,推荐从文件中读取私钥字符串而不是写入源码中
        config.merchantPrivateKey = this.privateKey;

        //注:如果采用非证书模式,则无需赋值上面的三个证书路径,改为赋值如下的支付宝公钥字符串即可
        config.alipayPublicKey = this.publicKey;

        //可设置异步通知接收服务地址(可选)
        config.notifyUrl = notifyUrl;

        return config;
    }
}

4. 唤起支付

创建一个接口,调用唤起支付宝支付,注意:唤起支付页面可能存在多种情况,

  1. 当面付 FaceToFace
  2. 电脑网站 Page
  3. 手机网站 Wap
  4. App支付 App
    实际是哪一个支付环境使用对应的方法,具体的方法参数等参考支付宝服务端API的详细参数说明

代码:

/**
 * <p>
 * 前端控制器
 * </p>
 *
 * @author 
 * @since 2021-05-18
 */
@RestController
@CrossOrigin
@RequestMapping("/card")
public class CardController {
    @Autowired
    CardService cardService;
    @Autowired
    CardRecordService cardRecordService;
    //支付成功后要跳转的页面
    @Value("${alipay.returnUrl}")
    private String returnUrl;

    // private Map<String, String> map = new HashMap<>();

    /**
     * 充值
     */
    @PostMapping(value = "/addMoney", produces = {"text/html;charset=UTF-8"})
    public Object add(CardRecord cardRecord) throws Exception {
        cardRecord.setOrderNo(OrderUtil.getOrderNo());
        cardRecord.setType("1");
        cardRecordService.save(cardRecord);
        //手机网站
        // 参数一:订单名
        //参数二:订单号
        //参数三:金额
        //参数四:用户付款中途退出返回商户网站的地址
        //参数五:返回的url
        //AlipayTradeWapPayResponse response = Factory.Payment.Wap().pay("充值", cardRecord.getOrderNo(), cardRecord.getMoney().toString(), "", returnUrl);
        //电脑网站支付
        // 参数一:订单名
        //参数二:订单号
        //参数三:金额
        //参数四:返回的url
        AlipayTradePagePayResponse response = Factory.Payment.Page().pay("充值", cardRecord.getOrderNo(), cardRecord.getMoney().toString(), returnUrl);
        return response.body;
    }

    /**
     * 充值成功进行的回调,根据自己业务逻辑
     * 唤起支付的参数orderNo会在回调中返回,获取到订单号关联出订单。
     */
    @PostMapping("/fallMoney")
    public Result add(HttpServletRequest request) {
        //支付宝回调参数
        Map<String, String> map = new HashMap<>();
        Enumeration<String> paramsName = request.getParameterNames();
        while (paramsName.hasMoreElements()) {
            String key = paramsName.nextElement();
            String value = request.getParameter(key);
            map.put(key, value);
            System.out.println("接收到的参数---->" + key + "/values:" + value);
        }
        //接收到订单号和金额
        String orderNo = String.valueOf(map.get("out_trade_no"));
        String money = String.valueOf(map.get("buyer_pay_amount"));
        BigDecimal money1 = new BigDecimal(money);
        System.out.println("接收到的参数订单金额---->" + map.get("buyer_pay_amount") + "/订单号" + map.get("out_trade_no"));
        //根据订单号查询carRecord并修改支付状态
        CardRecord cardRecord1 = cardRecordService.getOne(Wrappers.<CardRecord>lambdaQuery().eq(CardRecord::getOrderNo, orderNo));
        cardRecord1.setPayStatus("1");
        cardRecordService.updateById(cardRecord1);
        //获取到当前卡和余额
        QueryWrapper wrapper = new QueryWrapper();
        wrapper.eq("user_id", cardRecord1.getUserId());
        Card card = cardService.getOne(wrapper);
        System.out.println("当前用户卡信息---->" + card);
        if (card == null) {
            return Result.fail("数据异常");
        }
        //更新卡余额
        card.setBalance(card.getBalance().add(money1));
        cardService.updateById(card);
        return Result.success(null);
    }

方案二

1. 导入依赖

<!--aliPay-->
<dependency>
    <groupId>com.alipay.sdk</groupId>
    <artifactId>alipay-easysdk</artifactId>
    <version>2.2.0</version>
</dependency>

2. 唤起支付

/**
 * <p>
 * 前端控制器
 * </p>
 *
 * @author 
 * @since 2021-05-18
 */
@RestController
@CrossOrigin
@RequestMapping("/card")
public class CardController {
    @Autowired
    CardService cardService;
    @Autowired
    CardRecordService cardRecordService;
    //支付成功后要跳转的页面
    @Value("${alipay.returnUrl}")
    private String returnUrl;

    // private Map<String, String> map = new HashMap<>();

    /**
     * 充值
     */
    @PostMapping(value = "/addMoney", produces = {"text/html;charset=UTF-8"})
    public Object add(CardRecord cardRecord) throws Exception {
        cardRecord.setOrderNo(OrderUtil.getOrderNo());
        cardRecord.setType("1");
        cardRecordService.save(cardRecord);
        //写法一:不封装,写死的参数
        MultipleFactory multipleFactory=new MultipleFactory();
        multipleFactory.setOptions(getOptions());
        AlipayTradePagePayResponse response = multipleFactory.Page()
                //.agent()这个参数是指定appAuthToken
                //.agent("")
                //调用asyncNotify扩展方法,可以为每次API调用,设置独立的异步通知地址,此处设置的异步通知地址的优先级高于全局Config中配置的异步通知地址
                .asyncNotify(returnUrl)
                //参数一:订单名 参数二:订单号 参数三:金额 参数四:返回的url
                .pay("充值", cardRecord.getOrderNo(), cardRecord.getMoney().toString(), returnUrl);
        //写法二:封装,根据店铺id获取
        MultipleFactory multipleFactory = AliPayConfiguration.getFactory(cardRecord.getShopId);
        AlipayTradePagePayResponse response = multipleFactory.Page()
                //.agent()这个参数是指定appAuthToken
                //.agent("")
                //调用asyncNotify扩展方法,可以为每次API调用,设置独立的异步通知地址,此处设置的异步通知地址的优先级高于全局Config中配置的异步通知地址
                .asyncNotify(returnUrl)
                //参数一:订单名 参数二:订单号 参数三:金额 参数四:返回的url
                .pay("充值", cardRecord.getOrderNo(), cardRecord.getMoney().toString(), returnUrl);
        return response.body;
    }
    /**
     * 充值成功进行的回调
     */
    @PostMapping("/fallMoney")
    public Result add(HttpServletRequest request) {
        //支付宝回调参数
        Map<String, String> map = new HashMap<>();
        Enumeration<String> paramsName = request.getParameterNames();
        while (paramsName.hasMoreElements()) {
            String key = paramsName.nextElement();
            String value = request.getParameter(key);
            map.put(key, value);
            System.out.println("接收到的参数---->" + key + "/values:" + value);
        }
        //接收到订单号和金额
        String orderNo = String.valueOf(map.get("out_trade_no"));
        String money = String.valueOf(map.get("buyer_pay_amount"));
        BigDecimal money1 = new BigDecimal(money);
        System.out.println("接收到的参数订单金额---->" + map.get("buyer_pay_amount") + "/订单号" + map.get("out_trade_no"));
        //根据订单号查询carRecord并修改支付状态
        CardRecord cardRecord1 = cardRecordService.getOne(Wrappers.<CardRecord>lambdaQuery().eq(CardRecord::getOrderNo, orderNo));
        cardRecord1.setPayStatus("1");
        cardRecordService.updateById(cardRecord1);
        //获取到当前卡和余额
        QueryWrapper wrapper = new QueryWrapper();
        wrapper.eq("user_id", cardRecord1.getUserId());
        Card card = cardService.getOne(wrapper);
        System.out.println("当前用户卡信息---->" + card);
        if (card == null) {
            return Result.fail("数据异常");
        }
        //更新卡余额
        card.setBalance(card.getBalance().add(money1));
        cardService.updateById(card);
        return Result.success(null);
    }
    private Config getOptions() {

        //这里省略了一些不必要的配置,可参考文档的说明

        Config config = new Config();
        config.protocol = "https";
        config.gatewayHost = "openapi.alipaydev.com";
        config.signType = "RSA2";

        config.appId = "参数自备";

        // 为避免私钥随源码泄露,推荐从文件中读取私钥字符串而不是写入源码中
        config.merchantPrivateKey = "参数自备";

        //注:如果采用非证书模式,则无需赋值上面的三个证书路径,改为赋值如下的支付宝公钥字符串即可
        config.alipayPublicKey = "参数自备";

        //可设置异步通知接收服务地址(可选)
        config.notifyUrl = "http://eh6jnw.natappfree.cc/card/fallMoney";

        return config;
    }

附上写法二的AliPayConfiguration:

package com.mallshop.config;

import com.alipay.easysdk.factory.MultipleFactory;
import com.alipay.easysdk.kernel.Config;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;

/**
 * 获取支付
 * @author lenovo
 * @date 2023/2/1
 */
@Slf4j
@Configuration
public class AliPayConfiguration {

    private static PayConfigService payConfigService;

    @Autowired
    public AliPayConfiguration(PayConfigService payConfigService) {
        this.payConfigService = payConfigService;
    }

    /**
     * 获取Factory
     * @return
     */
    public static MultipleFactory getFactory(String shopId) throws Exception {
        MultipleFactory multipleFactory = new MultipleFactory();
        multipleFactory.setOptions(getOptions(shopId));
        return multipleFactory;
    }

    private static Config getOptions(String shopId) throws Exception {
        PayConfig payConfig = payConfigService.getOne(Wrappers.<PayConfig>query().lambda()
                .eq(PayConfig::getShopId, shopId));
        if(payConfig == null){
            throw new Exception("当前店铺没有配置支付宝");
        }
        Config config = new Config();
        config.protocol = "https";
        config.gatewayHost = "openapi.alipay.com";
        config.signType = "RSA2";
        config.appId = payConfig.getAppId();

        // 为避免私钥随源码泄露,推荐从文件中读取私钥字符串而不是写入源码中
        config.merchantPrivateKey = payConfig.getMchKey();

        //注:证书文件路径支持设置为文件系统中的路径或CLASS_PATH中的路径,优先从文件系统中加载,加载失败后会继续尝试从CLASS_PATH中加载
        //请填写您的应用公钥证书文件路径,
        config.merchantCertPath = payConfig.getPrivateCertPath();
        //请填写您的支付宝公钥证书文件路径,
        config.alipayCertPath = payConfig.getPrivateKeyPath();
        //请填写您的支付宝根证书文件路径,
        config.alipayRootCertPath = payConfig.getKeyPath();

        //注:如果采用非证书模式,则无需赋值上面的三个证书路径,改为赋值如下的支付宝公钥字符串即可
        // config.alipayPublicKey = "<-- 请填写您的支付宝公钥,例如:MIIBIjANBg... -->";

        return config;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/193931.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Micropython ESP32

Micropython ESP32模块列表network模块WIFI STA模式WIFI AP模式machine模块CPU主频GPIO端口GPIO输入模式GPIO输出模式GPIO中断模式ADC模数转换DAC数模转换PWM脉冲宽度调制UART串口Timer定时器官方文档 下载固件 模块列表 network模块 help(network) object <module ‘net…

域名基础知识

1.域名的概念及作用 域名&#xff08;Domain Name&#xff09;&#xff0c;又称网域&#xff0c;是由一串用点分隔的名字组成的Internet上某一台计算机或计算机组的名称&#xff0c;用于在数据传输时对计算机的定位标识&#xff08;有时也指地理位置&#xff09;。 由于IP地址…

vulnhub之PRIME (2021): 2

1.信息收集 输入arp-scan 192.168.239.0/24发现192.168.239.168主机存活。 使用nmap对目标主机192.168.239.168进行端口收集,&#xff0c;发现存活端口&#xff1a;22、80、139、445、10123。 访问http://192.168.239.168/&#xff0c;没有发现可用的信息。 使用gobuster进…

1、Maven——Maven项目管理工具基本设置、把Maven集成到IDEA2022

目录 一、Maven相关参数配置 1、配置依赖&#xff08;jar包&#xff09;存储位置&#xff08;本地仓库&#xff09; 2、 配置依赖下载地址 二、把Maven集成到IDEA2022 一、Maven相关参数配置 1、配置依赖&#xff08;jar包&#xff09;存储位置&#xff08;本地仓库&#…

vue使用echarts 仪表盘样式不对 | 使用echarts5.0

最近在使用Echarts官网样例的仪表盘图时候发现自己用的和官网的样例样式完全不一样。 无论怎么调整参数都还是没有办法解决。如果有同学碰到和我一样的问题可以尝试一下使用最新版的Echarts&#xff08;5.0以上&#xff09;。 因为曾经也怀疑过Echarts版本问题因此npm install…

MySQL详解(五)——高级 3.0

查询截取分析 慢查询日志 MySQL的慢查询日志是MySQL提供的一种日志记录&#xff0c;它用来记录在MySQL中响应时间超过阀值的语句&#xff0c;具体指运行时间超过long_query_time值的SQL&#xff0c;则会被记录到慢查询日志中。 具体指运行时间超过long_query_time值的SQL&am…

汇编语言-实现一个简单的主引导记录(MBR)引导用户程序

本文参考李忠老师的《X86汇编语言&#xff1a;实模式到保护模式》 前言 自己手动实现一个简单的主引导记录来引导用户程序&#xff0c;有助于了解 主引导程序的工作流程在汇编代码层面如何调用函数&#xff08;函数调用的原理&#xff09;在汇编代码层面如何读写硬盘&#xf…

Android中级——滑动分析

SrcollAndroid坐标系视图坐标系常见方法实现滑动layout()offsetLeftAndRight()和offsetTopAndBottom()LayoutParamsscrollTo()与scrollBy()ScrollerVierDragHeplerAndroid坐标系 将屏幕左上角的顶点作为Android坐标系的原点&#xff0c;向右为X轴正方向&#xff0c;向下为Y轴正…

uni-app中uni-ui组件库的使用

介绍uni-ui是DCloud提供的一个跨端ui库&#xff0c;它是基于vue组件的、flex布局的、无dom的跨全端ui框架。uni-ui不包括基础组件&#xff0c;它是基础组件的补充特点高性能&#xff08;自动差量更新数据&#xff0c;优化逻辑层和视图层通讯折损&#xff0c;背景停止&#xff0…

Leetcode力扣秋招刷题路-0337

从0开始的秋招刷题路&#xff0c;记录下所刷每道题的题解&#xff0c;帮助自己回顾总结 337. 打家劫舍 III&#xff08;Mid&#xff09; 小偷又发现了一个新的可行窃的地区。这个地区只有一个入口&#xff0c;我们称之为 root 。 除了 root 之外&#xff0c;每栋房子有且只有一…

ESP32+Arduino+OLED+u8g2播放视频

1、思路分析 ESP32采用Arduino开发&#xff0c;结合u8g2模块可以很方便地实现在oled上显示图片。因此&#xff0c;只需要将一个视频拆开成一帧帧&#xff0c;然后循环显示即可。 然而&#xff0c;有几个问题&#xff1a; 视频太大&#xff0c;esp32的flash无法存下怎么办&…

DynaSLAM-8 DynaSLAM中双目运行流程(Ⅱ):初始化SLAM系统部分System.cc

目录 1.回忆 2.System::System 1.回忆 上篇博客中我们讲述了DynaSLAM中初始化Mask R-CNN网络部分的代码。 这篇博客我们讲述初始化DynaSLAM除Mask R-CNN网络部分以外的代码。 2.System::System 初始化Mask R-CNN网络后&#xff0c;程序执行&#xff1a; // Create SLAM syst…

MongoDB 4.0支持事务了,还有多少人想用MySQL呢?

目录一、MongoDB 不支持事务&#xff1f;二、什么是事务&#xff1f;三、ACID的定义四、如何使用事务五、重要参数简介1、时间限制2、oplog大小限制六、连接池 数据库连接的缓存1、MongoDB查询数据五步走2、MongoDB连接池的参数配置七、聚合框架八、MongoDB文档格式设计1、限制…

【八大数据排序法】插入排序法的图形理解和案例实现 | C++

第十六章 插入排序法 目录 第十六章 插入排序法 ●前言 ●认识算法 ●一、插入排序法是什么&#xff1f; 1.简要介绍 2.图形理解 3.算法分析 ●二、案例实现 1.案例一 ●总结 前言 排序算法是我们在程序设计中经常见到和使用的一种算法&#xff0c;它主要是将…

MySQL【left join、right join、inner join】详细用法

参考链接&#xff1a;mysql的left join和inner join的详细用法https://blog.csdn.net/weixin_45906830/article/details/111133181 1. inner join&#xff1a;内连接&#xff1a;显示两个表中有联系的所有数据。 通俗讲&#xff1a;inner join 查找的数据是左右两张表共有的。 …

【C语言练习】字符串旋转你会嘛?

目录&#x1f36c;题目描述&#xff1a;&#x1f36d;思路一&#xff1a;&#x1f361;代码优化&#xff1a;&#x1f36d;思路二&#xff1a;&#x1f36c;题目描述&#xff1a;&#x1f36d;思路一&#xff1a;&#x1f36d;思路二&#xff1a;&#x1f36c;题目描述&#xf…

车辆控制器的 Fail Safe功能介绍

Fail Safe概要 在漆黑的夜路上&#xff0c;一辆开着头灯的汽车经过。 如果控制前照灯的控制器在这种情况下发生故障怎么办&#xff1f; 大灯会熄灭&#xff0c;造成危险吗&#xff1f; 不。 在这种情况下&#xff0c;控制器的“Fail Safe”被激活&#xff0c;前照灯保持其先前的…

企业需要一个数字体验平台(DXP)吗?

数字体验平台是一个软件框架&#xff0c;通过与不同的业务系统喝解决方案集成&#xff0c;帮助企业和机构建立、管理和优化跨渠道的数字体验。帮助企业实现跨网站、电子邮件、移动应用、社交平台、电子商务站点、物联网设备、数字标牌、POS系统等传播内容&#xff0c;除了为其中…

termux入门安装

下载安装 请使用F-Droid 的Termux&#xff0c;GooglePlay的 Termux 可能存在一些问题。 下载地址&#xff1a;https://f-droid.org/en/packages/com.termux/ 下载完成在安卓手机上直接安装Termux的apk文件就可以了。 termux换源 新版本的termux换源一条命令就可以超简单&…

【C++之类和对象】初识类和对象

目录前言一、面向对象VS面向过程二、类三、类的定义四、类的访问限定符五、封装六、C中的用struct和用class定义的类有何不同&#xff1f;七、类的作用域八、类的实例化九、计算类对象的大小十、this指针前言 C是一门面向对象的语言&#xff0c;之前学习的C语言是一种面向过程的…