springboot 微信小程序支付

2023-11-08

简单对支付封装,
使用到maven 依赖,版本依照自己项目情况自行添加

	<dependency>
        <groupId>com.alibaba</groupId>
        <artifactId>fastjson</artifactId>
    </dependency>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
    </dependency>
	<dependency>
        <groupId>io.springfox</groupId>
        <artifactId>springfox-swagger2</artifactId>
	</dependency>

1、请求参数实体基类,
因用于微信app支付、小程序支付、h5支付
首先创建基类BasePayEntity 实现Serializable 接口



@Data
@NoArgsConstructor
@Accessors(chain = true)
public class BasePayEntity implements Serializable {

    @JSONField(name = "appid")
    @ApiModelProperty(name = "appId"        , required = true , value = "应用ID")
    protected String appId ;

    @JSONField(name = "mchid")
    @ApiModelProperty(name = "mchId"        , required = true , value = "直连商户号")
    protected String mchId ;

    @JSONField(name = "description")
    @ApiModelProperty(name = "description"  , required = true , value = "商品描述")
    protected String description ;

    @JSONField(name = "out_trade_no")
    @ApiModelProperty(name = "outTradeNo"   , required = true , value = "商户订单号")
    protected String outTradeNo ;

    @JSONField(name = "time_expire")
    @ApiModelProperty(name = "timeExpire"                     , value = "交易结束时间")
    protected String timeExpire ;

    @JSONField(name = "attach")
    @ApiModelProperty(name = "attach"                         , value = "附加数据")
    protected String attach ;

    @JSONField(name = "notify_url")
    @ApiModelProperty(name = "notifyUrl"    , required = true , value = "通知地址")
    protected String notifyUrl ;

    @JSONField(name = "amount")
    @ApiModelProperty(name = "amount"       , required = true , value = "订单金额")
    protected Amount amount ;

    @JSONField(name = "detail")
    @ApiModelProperty(name = "detail"                         , value = "优惠功能")
    protected Detail detail ;

    @JSONField(name = "settle_info")
    @ApiModelProperty(name = "settleInfo"                      , value = "结算信息")
    protected SettleInfo settleInfo ;

    @Data
    @Builder
    @NoArgsConstructor
    @AllArgsConstructor
    public static class Amount implements Serializable {

        @ApiModelProperty(name = "total"    , required = true  , value = "总金额")
        protected Integer total ;

        @ApiModelProperty(name = "currency" , required = true  , value = "CNY:人民币,境内商户号仅支持人民币。")
        protected String currency ;

    }

    @Data
    @Builder
    @NoArgsConstructor
    @AllArgsConstructor
    public static class Detail implements Serializable{

        @JSONField(name = "cost_price")
        @ApiModelProperty(name = "costPrice"                  , value = "总金额")
        protected Integer costPrice ;

        @JSONField(name = "invoice_id")
        @ApiModelProperty(name = "invoiceId"                  , value = "商品小票ID")
        protected String invoiceId ;

    }



    @Data
    @Builder
    @NoArgsConstructor
    @AllArgsConstructor
    public static class SettleInfo implements Serializable{

        @JSONField(name = "profit_sharing")
        @ApiModelProperty(name = "profitSharing"              , value = "是否指定分账")
        protected Boolean profitSharing ;

    }

    public BasePayEntity(String appId, String mchId, String description, String outTradeNo, String timeExpire, String attach, String notifyUrl, Amount amount, Detail detail,  SettleInfo settleInfo) {
        this.appId = appId;
        this.mchId = mchId;
        this.description = description;
        this.outTradeNo = outTradeNo;
        this.timeExpire = timeExpire;
        this.attach = attach;
        this.notifyUrl = notifyUrl;
        this.amount = amount;
        this.detail = detail;
        this.settleInfo = settleInfo;
    }
}

2、定义Jsapi下单
创建实体类继承BasePayEntity



@Data
@NoArgsConstructor
@AllArgsConstructor
@ApiModel("Jsapi下单")
@Accessors(chain = true)
@Url("https://api.mch.weixin.qq.com/v3/pay/transactions/jsapi")
public class JsApiPay extends BasePayEntity {


    @JSONField(name = "scene_info")
    @ApiModelProperty(name = "sceneInfo"                     , value = "支付场景描述")
    private SceneInfo sceneInfo ;

    @JSONField(name = "payer")
    @ApiModelProperty(name = "payer"       , required = true , value = "支付者信息")
    private Payer payer ;


    @Data
    @Builder
    @NoArgsConstructor
    @AllArgsConstructor
    public static class Payer implements Serializable{

        @ApiModelProperty(name = "openid"  , required = true , value = "用户标识")
        private String openid ;

    }

    @Data
    @Builder
    @NoArgsConstructor
    @AllArgsConstructor
    public static class SceneInfo implements Serializable{
        @JSONField(name = "payer_client_ip")
        @ApiModelProperty(name = "payerClientIp"        , value = "用户的客户端IP,支持IPv4和IPv6两种格式的IP地址")
        private String payerClientIp ;

        @JSONField(name = "device_id")
        @ApiModelProperty(name = "deviceId"             , value = "商户端设备号(门店号或收银设备ID)")
        private String deviceId ;

        @JSONField(name = "store_info")
        @ApiModelProperty(name = "storeInfo"            , value = "商户门店信息")
        private StoreInfo storeInfo ;


        @Data
        @Builder
        @NoArgsConstructor
        @AllArgsConstructor
        public static class StoreInfo implements Serializable{
            @JSONField(name = "id")
            @ApiModelProperty(name = "id"           , value = "商户侧门店编号")
            private String id ;

            @JSONField(name = "name")
            @ApiModelProperty(name = "name"         , value = "商户侧门店名称")
            private String name ;

            @JSONField(name = "area_code")
            @ApiModelProperty(name = "areaCode"     , value = "地区编码,详细请见省市区编号对照表。")
            private String areaCode ;

            @JSONField(name = "address")
            @ApiModelProperty(name = "address"      , value = "详细的商户门店地址")
            private String address ;
        }

    }

    @Builder(toBuilder = true)
    public JsApiPay(String appId, String mchId, String description, String outTradeNo, String timeExpire, String attach, String notifyUrl, Amount amount, Detail detail, SettleInfo settleInfo, SceneInfo sceneInfo, Payer payer) {
        super(appId, mchId, description, outTradeNo, timeExpire, attach, notifyUrl, amount, detail, settleInfo);
        this.sceneInfo = sceneInfo;
        this.payer = payer;
    }
}

3、创建工具MiniPayUtils




public class MiniPayUtils {

    private WechatProperties wechatProperties ;


    public static MiniPayUtils newInstance(WechatProperties wechatProperties){
        MiniPayUtils miniPayUtils = new MiniPayUtils();
        miniPayUtils.wechatProperties = wechatProperties ;
        return miniPayUtils ;
    }


    public MiniPayResult pay(JsApiPay jsApiPay) {
        BasePayEntity.Amount amount = jsApiPay.getAmount();
        jsApiPay = jsApiPay.toBuilder().appId(wechatProperties.getMini().getAppid()).amount(ObjectUtils.isEmpty(amount) ? BasePayEntity.Amount.builder().currency("CNY").build() : amount.setCurrency("CNY")).notifyUrl(wechatProperties.getNotifyUrl()).mchId(wechatProperties.getMerchant().getMchId()).build();

        String nonceStr = SignUtils.getRandom() , timeStamp = String.valueOf(System.currentTimeMillis() / 1000) ;
        MiniPayResult miniPayResult = MiniPayResult.builder().appId(jsApiPay.getAppId())
                                                             .timeStamp(timeStamp)
                                                             .nonceStr(nonceStr)
                                                             .packages(prepayId(jsApiPay , nonceStr , timeStamp))
                                                             .signType("RSA")
                                                             .build();
        miniPayResult.setPaySign(SignUtils.sign(miniPayResult.toJsonString().getBytes() , wechatProperties.getMerchant().getCertPath() , ProjectStatus.windows操作系统.getValue().equals(wechatProperties.getEnvironment())));
        return miniPayResult ;
    }



 public String prepayId(JsApiPay jsapiPay , String nonceStr , String timeStamp){
        try{
            String url = jsapiPay.getClass().getAnnotation(Url.class).value() , body = JSONObject.toJSONString(jsapiPay);
            HttpHeaders httpHeaders = new HttpHeaders();
            httpHeaders.add(HttpHeaders.AUTHORIZATION , SignUtils.getToken(HttpMethod.POST.name()
                                                                          , new URL(url)
                                                                          , body
                                                                          , nonceStr
                                                                          , timeStamp
                                                                          , jsapiPay.getMchId()
                                                                          , wechatProperties.getMerchant().getSerialNo()
                                                                          , wechatProperties.getMerchant().getCertPath()
                                                                          , ProjectStatus.windows操作系统.getValue().equals(wechatProperties.getEnvironment())));
            httpHeaders.add(HttpHeaders.CONTENT_TYPE  , MediaType.APPLICATION_JSON_VALUE);
            ResponseEntity<String> exchange = getRestTemplate().exchange(url , HttpMethod.POST , new HttpEntity(body , httpHeaders), String.class );
            if(HttpStatus.OK.equals(exchange.getStatusCode())) {
                JSONObject jsonObject = JSONObject.parseObject(exchange.getBody());
                return "prepay_id=" + jsonObject.getString("prepay_id");
            }
            throw BaseException.throwEx(JSONObject.parseObject(exchange.getBody()).getString("message"));
        }catch (Exception e){
            String message = e.getMessage();
            if(e instanceof HttpClientErrorException){
                HttpClientErrorException exception = (HttpClientErrorException)e;
                if(403 == exception.getRawStatusCode()) throw BaseException.throwEx(JSONObject.parseObject(message.substring(message.indexOf("[") + 1 , message.lastIndexOf("]"))).getString("message"));
            }
            throw BaseException.throwEx(message);
        }
    }


}
以上的代码基本上可以完成项目的微信支付
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

springboot 微信小程序支付 的相关文章

随机推荐

  • vue 数组按时间排序

  • 一文理解Kafka如何保证消息顺序性

    要想实现消息有序 需要从Producer和Consumer两方面来考虑 如果对Kafka不了解的话 可以先看这篇博客 一文快速了解Kafka 针对消息有序的业务需求 还分为全局有序和局部有序 全局有序 一个Topic下的所有消息都需要按照生
  • 用spss做mk检验_SPSS中对问卷数据进行探索因子分析的详细操作与结果解读

    一 什么是探索因子分析 探索因子分析主要用于对问卷数据进行统计分析 通常我们在做问卷分析的时候 如果我们设计的题项没有明确的维度划分 而我们又需要了解这些题项的维度 这时我们就需要对这题项进行探索因子分析 通过在SPSS中进行探索因子分析
  • Docker世界 -- 基础篇(入门)

    一 Docker概述 1 1 Docker 为什么会出现 一款产品从开发到上线 从操作系统 到运行环境 在到应用配置 作为开发 运维之间的协作 我们需要关心很多东西 这也是很多互联网公司都不得不面对的问题 特别是各种版本的迭代之后 不同版本
  • Python 3 安装 redis 客户端

    1 去GIThub 上下载 redis py 2 然后进入源码目录 执行 sudo python3 setup py install 就可以了 这里踩了一个坑 就是我按照GITHub上介绍的方式 sudo python setup py i
  • 区块链矿工如何以七个步骤处理挖掘工作并进行交易

    你有没有想过区块链的挖掘过程如何进行的 或者你的交易如何得到确认并添加到了区块链 好吧 我也是如此 由于我找不到任何明确的逐步解释这个过程文章 我决定深入研究并自己编写指南 以下是区块链交易如何通过七个步骤从你的钱包处理到区块链中 imag
  • git 第一次push error: failed to push some refs to

    第一次将本地文件push到远程时报错 error failed to push some refs to 远程地址 原因可能是github仓库中的README md文件不在本地代码目录中或要先pull 遇到错误要善用git的提示 报错信息后
  • yolov7 paper阅读笔记

    不同于现有的主流方法 yolov7的主要改进点在优化训练过程 包括优化modules 和 优化方法 这些会加强traing cost从而提高object detect的accuracy 但是不会提高inference cost 时间开销 这
  • calibrate_camera函数示例

    好的 下面是一个使用 Python 语言的 calibrate camera 函数示例 import numpy asnp import cv2 def calibrate camera images pattern size 准备对象点和
  • OSSEC直接向GMAIL发送alert的配制方法

    按照mannul的介绍 直接在ossec conf中配置如下
  • 手把手教你使用TensorFlow训练出自己的模型

    手把手教你使用TensorFlow训练出自己的模型 一 前言 搭建TensorFlow开发环境一直是初学者头疼的问题 为了帮忙初学者快速使用TensorFlow框架训练出自己的模型 作者开发了一款基于TensorFlow 2 3 0的图形化
  • java asynccallback_使用 AsyncCallback 处理异步调用

    参考文章 原文 异步调用可以避免主线程受工作线程阻塞 即工作线程执行的过程中 主线程依然可以往下运行 不必等待工作线程完成 下面是一个简单的异步调用加法函数的例子 using System using System Threading th
  • 肺部ct重建_肺部CT血管分割及三维重建

    摘要 现代社会人们的生活水平不断改善 由于膳食不合理以及锻炼的缺乏 我国血管类疾病发生率不断攀高 传统的逐张读片的诊断方式效率低下且依赖于医生个人的知识储备 本文以肺部CT序列为着重点 对其中的血管结构进行分割 使用三维重建技术对分割结果进
  • Git入门与使用 (三) 使用GitHub进行代码托管的相关操作

    文章目录 一 前言 二 使用GitHub进行代码托管的相关操作 1 推送本地仓库内容至远程仓库 2 克隆远程仓库内容至本地仓库 3 邀请他人加入项目团队 4 拉取远程仓库修改的内容 5 解决协同开发时产生的冲突 6 跨团队协作开发 7 Gi
  • 线性代数 --- 矩阵求逆的4种方法

    线性代数 矩阵求逆的4种方法 写在最前面 在大多数情况下 我们学习线性代数的目的是为了求解线性方程组Ax b 而不是为了求A的逆 单就解方程而言 LU分解是最实用的算法 只需按照A LU gt Ax b LUx b gt Ly b 正向回代
  • Redhat6.5的云主机停在某一个状态,不能继续启动

    Redhat6 5的云主机停在某一个状态 不能继续启动 1 现象 云主机 不能启动 状态如图 2 分析及诊断 制作云主机快照 导入到virt manager环境 使用guestfish工具 移除了 ss5 尝试启动 启动成功了 移除后 et
  • Oracle中group by用法

    Oracle中group by用法 在select 语句中可以使用group by 子句将行划分成较小的组 一旦使用分组后select操作的对象变为各个分组后的数据 使用聚组函数返回的是每一个组的汇总信息 使用having子句 限制返回的结
  • sql用于判断的函数decode\case when then else语法

    简单的判断可以用decode函数 例如 如果字段a是null 值为0 如果是 值也为0 否则就是a的值 select decode a null 0 0 a from table 当需要有多个条件并列的时候 用case when then
  • 物联网技术和计算机网络技术,物联网技术下计算机网络技术专业建设探讨

    摘 要 物联网应用领域的高速扩张为高校人才培养带来新的机遇和挑战 本文论述了计算机网络技术专业开设物联网方向的必要性和可行性 研究结合学院行业资源及优势专业培养网络技术专业人才的新思路 提出融合多专业的以智慧交通 智慧物流等物联网应用为特色
  • springboot 微信小程序支付

    简单对支付封装 使用到maven 依赖 版本依照自己项目情况自行添加