微信支付系列文章

  1. 微信支付-java后端实现
  2. 微信支付-vue 前端实现

java demo: 下载地址文章底部

技术栈

  • Spring boot
  • java
  • XML (微信在http协议中数据传输方案)
  • MD5 签名

微信支付术语

  • openid (OpenID是公众号一对一对应用户身份的标识)
  • app_id (公众号id,登录微信公众号–开发–基本配置中获得;)
  • key (收款商户后台进行配置,登录微信商户平台–账户中心–API安全-设置秘钥,设置32位key值;)
  • mch_id (收款商家商户号;)
  • certPath (API证书, 登录微信商户平台–账户中心-API安全-下载证书)

后端流程

服务端需要的核心操作, 总共分为以下几步:

  1. 统一下单
  2. 前端调起微信支付必要参数 (需加密)
  3. 订单结果主动通知 (回调接口)
  4. 查询订单结果
  5. 结束订单支付接口(关闭订单,支付订单关闭)

代码

微信总共支持多种语言的sdk, 在官网可以下载例子, java程序也可以引入微信支付的sdk包, 但是github上的sdk已经很久没有更新了, 最好的选择, 也是我的选择, 在官网上下载sdk项目, 将其中所有java类copy到自己的项目中.

官网sdk下载目录
链接: 商户平台首页

#### 根据微信sdk生成配置类 WXPayConfig
创建IWxPayConfig.class, 继承sdk WXPayConfig.class, 实现sdk中部分抽象方法, 读取本地证书, 加载到配置类中.
package core.com.chidori.wxpay;

import core.com.wxpay.IWXPayDomain;
import core.com.wxpay.WXPayConfig;
import core.com.wxpay.WXPayConstants;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;@Service
public class IWxPayConfig extends WXPayConfig { // 继承sdk WXPayConfig 实现sdk中部分抽象方法private byte[] certData;@Value("${vendor.wx.config.app_id}")private String app_id;@Value("${vendor.wx.pay.key}")private String wx_pay_key;@Value("${vendor.wx.pay.mch_id}")private String wx_pay_mch_id;public IWxPayConfig() throws Exception { // 构造方法读取证书, 通过getCertStream 可以使sdk获取到证书String certPath = "/data/config/chidori/apiclient_cert.p12";File file = new File(certPath);InputStream certStream = new FileInputStream(file);this.certData = new byte[(int) file.length()];certStream.read(this.certData);certStream.close();}@Overridepublic String getAppID() {return app_id;}@Overridepublic String getMchID() {return wx_pay_mch_id;}@Overridepublic String getKey() {return wx_pay_key;}@Overridepublic InputStream getCertStream() {return new ByteArrayInputStream(this.certData);}@Overridepublic IWXPayDomain getWXPayDomain() { // 这个方法需要这样实现, 否则无法正常初始化WXPayIWXPayDomain iwxPayDomain = new IWXPayDomain() {@Overridepublic void report(String domain, long elapsedTimeMillis, Exception ex) {}@Overridepublic DomainInfo getDomain(WXPayConfig config) {return new IWXPayDomain.DomainInfo(WXPayConstants.DOMAIN_API, true);}};return iwxPayDomain;}
}

发起统一下单 AND 前端调起微信支付必要参数

// 发起微信支付
WXPay wxpay = null;
Map<String, String> result = new HashMap<>();
try {// ******************************************////  统一下单//// ******************************************wxpay = new WXPay(iWxPayConfig); // *** 注入自己实现的微信配置类, 创建WXPay核心类, WXPay 包括统一下单接口Map<String, String> data = new HashMap<String, String>();data.put("body", "订单详情");data.put("out_trade_no", transOrder.getGlobalOrderId()); // 订单唯一编号, 不允许重复data.put("total_fee", String.valueOf(transOrder.getOrderAmount().multiply(new BigDecimal(100)).intValue())); // 订单金额, 单位分data.put("spbill_create_ip", "192.168.31.166"); // 下单ipdata.put("openid", openId); // 微信公众号统一标示openiddata.put("notify_url", "http://wxlj.oopmind.com/payCallback"); // 订单结果通知, 微信主动回调此接口data.put("trade_type", "JSAPI"); // 固定填写logger.info("发起微信支付下单接口, request={}", data);Map<String, String> response = wxpay.unifiedOrder(data); // 微信sdk集成方法, 统一下单接口unifiedOrder, 此处请求   MD5加密   加密方式logger.info("微信支付下单成功, 返回值 response={}", response);String returnCode = response.get("return_code");if (!SUCCESS.equals(returnCode)) {return null;}String resultCode = response.get("result_code");if (!SUCCESS.equals(resultCode)) {return null;}String prepay_id = response.get("prepay_id");if (prepay_id == null) {return null;}// ******************************************////  前端调起微信支付必要参数//// ******************************************String packages = "prepay_id=" + prepay_id;Map<String, String> wxPayMap = new HashMap<String, String>();wxPayMap.put("appId", iWxPayConfig.getAppID());wxPayMap.put("timeStamp", String.valueOf(Utility.getCurrentTimeStamp()));wxPayMap.put("nonceStr", Utility.generateUUID());wxPayMap.put("package", packages);wxPayMap.put("signType", "MD5");// 加密串中包括 appId timeStamp nonceStr package signType 5个参数, 通过sdk WXPayUtil类加密, 注意, 此处使用  MD5加密  方式String sign = WXPayUtil.generateSignature(wxPayMap, iWxPayConfig.getKey());// ******************************************////  返回给前端调起微信支付的必要参数//// ******************************************result.put("prepay_id", prepay_id);result.put("sign", sign);result.putAll(wxPayMap);return result;
} catch (Exception e) {
}

回调结果处理

核心是支付订单回调时, 需校验加密签名是否匹配, 防止出现模拟成功通知

@RequestMapping(value = "/payCallback", method = RequestMethod.POST)
public String payCallback(HttpServletRequest request, HttpServletResponse response) {logger.info("进入微信支付异步通知");String resXml="";try{//InputStream is = request.getInputStream();//将InputStream转换成StringBufferedReader reader = new BufferedReader(new InputStreamReader(is));StringBuilder sb = new StringBuilder();String line = null;try {while ((line = reader.readLine()) != null) {sb.append(line + "\n");}} catch (IOException e) {e.printStackTrace();} finally {try {is.close();} catch (IOException e) {e.printStackTrace();}}resXml=sb.toString();logger.info("微信支付异步通知请求包: {}", resXml);return wxTicketService.payBack(resXml);}catch (Exception e){logger.error("微信支付回调通知失败",e);String result = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>" + "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";return result;}
}@Override
public String payBack(String notifyData) {logger.info("payBack() start, notifyData={}", notifyData);String xmlBack="";Map<String, String> notifyMap = null;try {WXPay wxpay = new WXPay(iWxPayConfig);notifyMap = WXPayUtil.xmlToMap(notifyData);         // 转换成mapif (wxpay.isPayResultNotifySignatureValid(notifyMap)) {// 签名正确// 进行处理。// 注意特殊情况:订单已经退款,但收到了支付结果成功的通知,不应把商户侧订单状态从退款改成支付成功String return_code = notifyMap.get("return_code");//状态String out_trade_no = notifyMap.get("out_trade_no");//订单号if (out_trade_no == null) {logger.info("微信支付回调失败订单号: {}", notifyMap);xmlBack = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>" + "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";return xmlBack;}// 业务逻辑处理 ****************************logger.info("微信支付回调成功订单号: {}", notifyMap);xmlBack = "<xml>" + "<return_code><![CDATA[SUCCESS]]></return_code>" + "<return_msg><![CDATA[SUCCESS]]></return_msg>" + "</xml> ";return xmlBack;} else {logger.error("微信支付回调通知签名错误");xmlBack = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>" + "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";return xmlBack;}} catch (Exception e) {logger.error("微信支付回调通知失败",e);xmlBack = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>" + "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";}return xmlBack;
}

注意点

  1. 统一下单的签名和后续前端拉取微信支付的签名需要统一, 也就是都采用MD5加密, 如果2者不同, 会导致前端拉取微信支付fail, 这是一个巨大的坑, 因为这个原因调试了好久, 微信在文档里没有明确标出统一下单的签名校验方式 需要和前端拉取微信支付的签名校验保持一致.
    微信sdk里的源码需要针对这个问题调整一下, 调整如下:
    WXPay类需要修改下加密判断,在WXPay构造方法中,调整如下

    public WXPay(final WXPayConfig config, final String notifyUrl, final boolean autoReport, final boolean useSandbox) throws Exception {this.config = config;this.notifyUrl = notifyUrl;this.autoReport = autoReport;this.useSandbox = useSandbox;if (useSandbox) {this.signType = SignType.MD5; // 沙箱环境}else {this.signType = SignType.MD5; // 将这里的加密方式修改为SignType.MD5, 保持跟前端吊起微信加密方式保持一致}this.wxPayRequest = new WXPayRequest(config);
    }
    

结束语

做完以后, 微信支付的后端逻辑还是很清晰的, 但是在开发过程中很煎熬, 不清楚每个专业术语在微信哪里配置, 加密方式乱的很

博客

请移步: https://www.oopmind.com。

Demo

git clone https://github.com/wangjianan1103/pica.git

联系方式

如果 Pica_pay 对你有帮助,可以关注作者支持一下,每天会不定时回复留言(有任何问题都可以留言哦)。

微信公众号

预览图

微信支付-java实现微信支付-后端篇相关推荐

  1. 手把手教你完成App支付JAVA后台-微信支付JAVA

    上篇我们记录了手机端的微信支付的大致流程,期间可能会遇到各种各样的错误,但这些问题没有得到官方的重视,所以我们只能一步步自己排查,要有足够的耐心. 这篇内容看标题已经很明确了,由于微信是用xml通讯的 ...

  2. 微信小程序支付-java对接微信

    一共是两个方法: 一个方法后台生成预支付订单,得到预支付交易会话标识prepay_id,传给前端,让前端调起小程序支付: 一个是支付回调 目录 一.生成预支付订单 注意: 二. 支付回调 一.生成预支 ...

  3. ping 支付 java代码_Ping++支付

    第一次接触支付啊,有点小激动,所以写下这篇随笔以防以后忘记. ping++的文档还有服务都是挺好的,当你注册之后,就会给你发邮件.截图如下: 是不是感觉服务很不错. 接下来直入正题. 首先,我们需要加 ...

  4. Java微信公众号高级 微信墙,JAVA折腾微信公众平台(Token验证)

    JAVA折腾微信公众平台(Token验证) JAVA折腾微信公众平台(Token验证) 2019独角兽企业重金招聘Python工程师标准>>> 最近微信的公众平台比较火,于是我也想弄 ...

  5. java微信支付代码_10行代码搞定微信支付(Java版)

    原标题:10行代码搞定微信支付(Java版) 微信支付痛点 对于大多数同学来说,要开发微信支付可不简单.附上微信支付官方文档网页链接 从文档上可以看出,你需要解决很多问题,我就随便挑几个吧. xml与 ...

  6. java获取微信accessToken

    java获取微信accessToken java获取微信accessToken GET请求 java获取微信accessToken package com.fengdi.lianmeng.task;i ...

  7. 扫码支付java,详解JAVA后端实现统一扫码支付:微信篇

    最近做完了一个项目,正好没事做,产品经理就给我安排了一个任务. 做一个像收钱吧这样可以统一扫码收钱的功能. 一开始并不知道是怎么实现的,咨询了好几个朋友,才知道大概的业务流程:先是开一个网页用来判断支 ...

  8. 微信小程序开发笔记 支付篇④——基于微信支付SDK实现Java后端接口使用

    文章目录 一.前文 二.微信支付 Java SDK 三.示例 一.前文 微信小程序开发笔记--导读 微信支付-SDK与DEMO下载 先看README.md 二.微信支付 Java SDK 对微信支付开 ...

  9. java对接支付宝微信银联_JavaWEB后端支付银联,支付宝,微信对接

    JavaWEB后端支付银联,支付宝,微信对接 标签(空格分隔): java 项目概述 最近项目需要后端打通支付,所以对接部分做成了一个小模块. 先说下项目要求: 后端要对接银联无跳转Token支付,支 ...

最新文章

  1. kobject_create_and_add
  2. Linux内核官方文档atomic_ops.txt【摘自Linux 内核文档】
  3. Leetcode周赛5193. 删除字符使字符串变好
  4. Maven实战 | dependencies与dependencyManagement
  5. Kali Linux 网络扫描秘籍 第六章 拒绝服务(二)
  6. 【状压DP】poj2686 Traveling by Stagecoach
  7. 【转载】【C基础】#define宏定义中的#,##,@#,\ 这些符号的神奇用法
  8. c++复习日记3 模板和流
  9. 回到顶部的几种实现方法
  10. 刨根究底字符编码之三——字符编码的由来
  11. 飞腾PHYTIUM FT-1500a性能测试-内存-PCIe
  12. C语言:递归实现Ackman函数
  13. 微信小程序开发(学习记录1.0)
  14. Android实现word模板套打功能
  15. android--GooglePay 谷歌支付内购接入(2)
  16. bzoj 1415 [Noi2005]聪聪和可可
  17. Python 之 列表推导式
  18. android 通过xmpp即时聊天客户端往服务器发消息,利用XMPP协议推送服务器告警信息到安卓平台及桌面...
  19. firefox安装java插件
  20. java-php-python-ssm面相高校学生的图书共享平台计算机毕业设计

热门文章

  1. RTL8211D(I)业级网口芯片bsp自适应修改
  2. Java程序控制结构、顺序控制、循环控制及跳转控制语句
  3. 华为新系统鸿蒙升级时间,好消息确定!鸿蒙之后华为全新系统发布,只期待升级尝鲜...
  4. 定制Eclipse的Content assist(代码补全),比如空格键不上屏
  5. 港澳联考数学可以用计算机吗,联考数学满分、北大学霸教你决胜港澳台联考
  6. MySQL8.0 驱动jar包
  7. 抖音小程序分享 onShareAppMessage 例子
  8. 初学c语言的方法和操作
  9. 74ls192/74ls193中文资料介绍-引脚图-真值表-工作原理
  10. PHP Web程序设计与Ajax技术pdf