jsapi支付签名_微信支付小程序支付全流程

点击蓝色字关注我们!

一个努力中的公众号

长的好看的人都关注了

cebef551aadc14e5ffd9bd2509cb9a12.png

本文给大家讲解微信小程序支付全流程,以及相关功能源代码,项目不开放,带来不便尽请谅解。小程序支付主要包含如下几步骤,1.预下单-调用微信统一下单接口进行预下单。2.小程序拿到支付参数唤醒支付。3.支付成功后发起退款申请本文使用okHttpUtil进行接口交互 使用goole的gson与阿里的fastJson进行解析JsonObect,
使用封装的http工具类进行携带证书进行接口交互
使用封装的工具类实现map转xml等。相关工具类代码过多,所以在本文最后有下载方式

下面是正文:

微信支付文档:  https://pay.weixin.qq.com/wiki/doc/api/wxa/wxa_api.php?chapter=9_1

一.获取微信用户的openId:用户登录小程序 通过授权得到微信code,然后将微信code、秘钥、密文传递后端,后端进行解密后 得到微信openId(前端也可以进行解密,但建议放后端)
下面是获取openId的代码(java)

//微信接口url 
https://api.weixin.qq.com/sns/jscode2session

/**
* 解析小程序登录数据获取参数* @param code ---微信code
* @param encryptedData --- 加密参数* @param iv --- 秘钥* @return*/

public Result> getAuthByApplet(String code, String encryptedData,     String iv) {

Map<String, Object> resultMap = new HashMap<>();
try {log.debug("=========小程序获取unionId=============: "+ code);//封装参数Map<String, String> params = new HashMap<String, String>();//获取小程序的openId

params.put("appid", appletId);

//微信秘钥

params.put("secret", appletSecret);params.put("js_code", code);params.put("grant_type", "authorization_code");//调用接口进行读取参数JsonObject retJson = OkHttpUtils.doGetRetJson(appletOpenIdUrl, params);log.debug(retJson.toString());//如果获取到了openIdif(retJson.get("openid") != null) {//获取会话密钥(session_key)String session_key = retJson.get("session_key").getAsString();resultMap.put("session_key", session_key);resultMap.put("openId", retJson.get("openid").getAsString());}else{throw new Exception("code已失效,请重新获取");}return Result.ok(resultMap);} catch (Exception e) {log.error(e.getMessage(), e);
throw new RuntimeException();}
}

得到openId后要进行业务保存,因为小程序很多接口都依赖openId,而且同一小程序下,每个微信号的openId是一样的。二:统一下单与小程序支付    通过业务系统的封装参数,得到支付的商品信息,价格,业务订单后,封装参数,进行加密得到sign签名,然后调用统一下单接口进行下单,成功后返回小程序支付参数。

(需要注意的是 微信预下单请求参数要求是xml格式,全String类型)

//微信预下单接口url
https://api.mch.weixin.qq.com/pay/unifiedorder

   /**
* 微信支付-统一下单* @param body --- 商品描述* @param outTradeNo --- 业务系统订单号* @param totalFee --- 支付金额* @param spBillCreateIP ---支付人所在
    * @param openId --- 支付人的微信openId

* @return

*/@Override

public Map<String, String> toPay(String body, String outTradeNo,              String totalFee, String spBillCreateIP, String openId) {

try{

Map<String, String> map = new HashMap<>();       //封装支付必须参数

SortedMap<Object,Object> params = new TreeMap<Object,Object>();

//支付类型 String tradeType = "JSAPI";

      //支付后回调地址

      String notifyUrl = "";

String nonceStr = RandomUtil.randomString(16);//小程序idparams.put("appid", appletId);//商户平台idparams.put("mch_id", mchId);//随机字符串params.put("nonce_str", nonceStr);//商品描述params.put("body", body);//商品订单号params.put("out_trade_no", outTradeNo);//价格的单位为分params.put("total_fee", totalFee);//支付ipparams.put("spbill_create_ip", spBillCreateIP);//异步回调apiparams.put("notify_url", notifyUrl);//JSAPI、h5调用params.put("trade_type", tradeType);//支付用户openidparams.put("openid", openId);String sign = WxPaySignatureUtils.createSign(params, key);params.put("sign", sign);String requestXML = WxPaySignatureUtils.getRequestXml(params);log.info("发送给微信的报文:" + requestXML);log.info("加密后的的签名字符串:" + sign);//调用微信接口String result = OkHttpUtils.postXml(unifiedorder, requestXML);log.info("-------------------请求返回结果------------------");log.info(result);map = WXPayUtil.xmlToMap(result);//如果返回有数据if(map != null){//并且返回有signif(map.get("sign") != null){return map;}
}return map;}catch (Exception e) {log.error(e.getMessage(), e);
throw new RuntimeException();}
}

接口请求成功后,微信返回也是xml格式的报文,需要将xml转为map或其他,本文使用map接收参数,预下单后,微信会返回预付订单标识,标识用于进支付

//预付订单标识String prepayId = resultMap.get("prepay_id");

然后封装小程序支付 所需参数

//签名类型,默认为MD5String signType = "MD5";SortedMap<Object,Object> params = new TreeMap<Object,Object>();params.put("appId", appletId);params.put("timeStamp", timestamp + "");params.put("timeStamp", timestamp + "");params.put("nonceStr", nonceStr);params.put("package", "prepay_id=" + prepayId);params.put("signType", signType);//生成签名String sign = WxPaySignatureUtils.createSign(params, key);params.put("paySign", sign);

将支付参数返回给小程序,小程序通过组件 requestPayment 唤醒支付小程序支付代码如下:

wx.requestPayment({

      timeStamp:param.result.timeStamp,

      nonceStr: param.result.nonceStr,

      package: param.result.package,

      signType:param.result.signType,

      paySign: param.result.paySign,

      success: function(res){

        console.log(res);

        wx.showToast({

          title: '支付成功',

          icon: 'success',

          duration: 2000

          });          

      },

      fail: function(res) {

        wx.showModal({

          title:'支付失败',

          content:'',

          showCancel: false

        })

      }

    })


三: 退款小程序支付成功后,可以通过支付订单号进行退款操作,需要注意的是,退款需要http携带商户证书进行支付,这里图图已经封装好工具类只需要修改下证书的url就可以愉快使用了

//微信退款使用的url
https://api.mch.weixin.qq.com/secapi/pay/refund

/**
* 微信支付-退款* @param transaction_id ---微信支付订单* @param out_refund_no --- 系统退款单号* @param total_fee --- 订单总金额* @param refund_fee --- 退款金额* @return*/@Overridepublic Map<String, String> refundToPay(String transaction_id, String out_refund_no, String out_trade_no, int total_fee, int refund_fee) {//定义返回参数Map<String, String> map = new HashMap<>();
try{SortedMap<Object,Object> params = new TreeMap<Object,Object>();String notifyUrl = "https://baidu.com";String nonceStr = RandomUtil.randomString(16);//小程序idparams.put("appid", appletId);//商户平台idparams.put("mch_id", mchId);//随机字符串params.put("nonce_str", nonceStr);//系统订单号params.put("out_trade_no", out_trade_no);//微信订单号params.put("transaction_id", transaction_id);//系统退款单号params.put("out_refund_no", out_refund_no);//订单金额params.put("total_fee", total_fee);//退款金额params.put("refund_fee", refund_fee);//签名String sign = WxPaySignatureUtils.createSign(params, key);params.put("sign", sign);//封装请求报文String requestXML = WxPaySignatureUtils.getRequestXml(params);log.info("发送给微信的报文:" + requestXML);log.info("加密后的的签名字符串:" + sign);//调用微信接口String result = WXPayUtil.doRefundRequest(mchId, refund, requestXML, certificateUrl);log.info("-------------------请求返回结果------------------");log.info(result);
if(StrUtil.isNotEmpty(result)){map = WXPayUtil.xmlToMap(result);//如果返回有数据if(map != null){//并且返回有signif(map.get("sign") != null){return map;}
}
}else{log.info("-------------------请求返回空结果------------------");}return map;}catch (Exception e) {log.error(e.getMessage(), e);
throw new RuntimeException();}
}

本文中所使用到的工具类,图图已经帮大家打包好了,一份在百度网盘,为防止百度网盘失效,在图图的QQ群中也有一份。

百度网盘下载链接

链接:https://pan.baidu.com/s/1YbMrvO3qK1rdmyYyET2xzw

提取码:ljv6

本文先总结到此,后面继续更新技术一些文章。大家一起加油鸭!

如果对您有帮助 请点个关注,万分感谢          

                                (QQ招聘群  710566091
                                 微信招聘群 请加图图微信)

d14b88b277eb91a947bbc8730d4ded2d.png

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

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

相关文章

攻击防护手段_如何防护ddos流量攻击?

量攻击分很多种&#xff1a;我们常见的DDOS CC SYN UDP等&#xff0c;目前在网络中&#xff0c;流量攻击比较难解决&#xff0c;普通的IDC机房或是服务器商都是依靠硬防来解决&#xff0c;但是碰到大流量攻击&#xff0c;往往是拔线了事&#xff0c;所以都不是什么好办法&#…

MyBatis参数名称解析器-ParamNameResolver解析

ParamNameResolver ParamNameResolver是一个参数名解析器&#xff0c;负责把方法的参数按顺序解析出来并进行标注 主要用来处理接口形式的参数&#xff0c;最后会把参数处放在一个map中 map的key为参数的位置&#xff0c;value为参数的名字 package org.apache.ibatis.refle…

json 取值判断_【收藏级】.NETCore3.1中的Json互操作解读

本文将会全面介绍System.Text.Json 和 Newtonsoft.Json 的相同和异同之处&#xff0c;方便需要的同学做迁移使用&#xff0c;对未来&#xff0c;我们保持期待。文档比较几个重要的对象在 System.Text.Json 中&#xff0c;有几个重量级的对象&#xff0c;所有的JSON互操作&#…

学习Java的深拷贝和浅拷贝

关于Java的深拷贝和浅拷贝&#xff0c;简单来说就是创建一个和已知对象一模一样的对象。可能日常编码过程中用的不多&#xff0c;但是这是一个面试经常会问的问题&#xff0c;而且了解深拷贝和浅拷贝的原理&#xff0c;对于Java中的所谓值传递或者引用传递将会有更深的理解。 …

转list_你知道Java数组转List的三种方式及对比吗?

前言&#xff1a;本文介绍Java中数组转为List三种情况的优劣对比&#xff0c;以及应用场景的对比&#xff0c;以及程序员常犯的类型转换错误原因解析。一.最常见方式(未必最佳)通过 Arrays.asList(strArray) 方式,将数组转换List后&#xff0c;不能对List增删&#xff0c;只能查…

Mybatis中的#号与$符号的区别

1、#{变量名}可以进行预编译、类型匹配等操作&#xff0c; 2、#{变量名}会转化为jdbc的类型。 3、${变量名}不进行数据类型匹配&#xff0c;直接替换。 4、#方式能够很大程度防止sql注入。 5、$方式无法方式sql注入。 6、$方式一般用于传入数据库对象&#xff0c;例如传入…

创建存储问题总结

创建存储过程时 1 drop procedure if exists pro_1; 2 delimiter // 3 create procedure pro_1( 4 5 ) 6 begin 7 8 end // 9 delimiter ; 发现这个问题时时因为看到输出的最后delimiter中有两个分号&#xff0c;所以怀疑这个地方语法存在问题 结尾的delimiter &#xff1b; …

Windows如何安装MSMQ消息队列

1 打开控制面板&#xff0c;找到下图所示的服务器核心&#xff0c;然点击确定 2 等待安装完成 转载于:https://www.cnblogs.com/acetaohai123/p/6610302.html

讲字节数组转化为base64_Base64编码简介及简单实现

Base64编码是一种将字节数据编码为字符串的编码&#xff0c;字节数据会被编码成由64个可打印ASCII字符组成的字符串,这64个字符包括大写字母A-Z, 小写字母a-z, 以及数字 0 -9再加上 和 / &#xff0c;刚好64个字符。对应的字符表如下图&#xff1a;base64编码的一个用途就是对…

Mybatis返回Map

返回一条记录的map resultType“map“ key就是列名&#xff0c;值就是对应的值 多条记录封装成一个map Select返回类型中是返回Map时&#xff0c;是对方法中是否存在注解MapKey&#xff0c;这个注解我也是第一次看到&#xff0c;当时我也以为是纯粹的返回单个数据对象的Map…

Mybatis中注解@MapKey的使用详解

MyBatis查询一些记录&#xff0c;数据涉及到两个表里的数据&#xff0c;需要连表查询&#xff0c;但我MyBatis的返回结果不想创建新的DO对象&#xff0c;因此使用MapKey注解返回一个Map集合。 含义&#xff1a;MapKey注解用于mapper.xml文件中&#xff0c;一般用于查询多条记录…

更新 绑定数据_Blazor 修仙之旅 组件与数据绑定

一.前言在第一篇文章初尝 Blazor WebAssembly中&#xff0c;有提到过组件(Component)这个概念。组件在 Blazor 中是必不可少的&#xff0c;UI 全靠它组装起来&#xff0c;和前端的 JS 组件是一个意思&#xff0c;比如&#xff1a;vue component、react component 等等。借用官方…

关于使用idea工具debug时,断点颜色由红色变成灰色解决方法

在使用断点调试的时候&#xff0c;发现断点由原来的红色变成灰色的&#xff0c;后来发现是由于错误操作将Debug断点调试禁用了 &#xff0c;只需要点击禁用按钮取消就可以了

改变图标颜色_LOL设计师宣布修改装备图标:提高清晰度、颜色差异化

在11月12日&#xff0c;英雄联盟更新到了最新的季前赛版本。这次官方除了对部分英雄进行改动&#xff0c;主要是对于装备的图标和属性进行了更新。而在图标更新后&#xff0c;很多装备的样子都发生了很大的变化。这也导致很多老玩家在进游戏后&#xff0c;看着装备栏发呆&#…

虚拟机中centos安装gcc

yum install gcc-c 一、首先是&#xff1a;使得虚拟机联网 使用NAT模式 虚拟机网络连接使用NAT模式&#xff0c;物理机网络连接使用Vmnet8。 虚拟机设置里面——网络适配器&#xff0c;网络连接选择NAT模式。 虚拟机菜单栏—编辑—虚拟网络编辑器&#xff0c;选择Vmnet8 NAT模式…

gcd(欧几里得算法)

基础 1 int gcd(int a,int b) 2 { 3 int r; 4 while(b>0) 5 { 6 ra%b; 7 ab; 8 br; 9 } 10 return a; 11 } View Code递归 1 int gcd(int a,int b) 2 { 3 return (b>0)?gcd(b,a%b):a; 4 } Vi…