• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

微信小程序 \"支付验证签名失败\" 问题 java 代码 ...

原作者: [db:作者] 来自: [db:来源] 收藏 邀请

最近在做一个微信小程序项目做到微信支付的时候遇到的一些问题!   

详细 步骤:

开发前准备(必须)

  小程序标识(appid):wx4d4838ebec29b8**

  商户号(mch_id):15508070**

  商户密钥(key) :wHtQckdfiRBVF7ceGTcSWEEORt6C0D**

我们用微信官方提供的SDK开发 : https://pay.weixin.qq.com/wiki/doc/api/native.php?chapter=11_1   

 下载 SDK 完成后 :  

 开始写我们的程序

  

进入微信支付 开发文档  : https://pay.weixin.qq.com/wiki/doc/api/index.html 

选择 小程序支付

 

选择 API列表  统一下单  可以看到微信 接口链接 和 请求参数 , 你需要看下每个参数什么意思,接下来就需要知道怎么操作这些参数就可以了  ok

 

一  首先 把刚下载的 微信提供的 SDK 拷贝到你的项目里    自定义一个类继承里面的一个  WXPayConfig 抽象类

 1 public class MyWxPayConfig extends WXPayConfig {
 2     private byte[] certData;
 3 
 4     public MyWxPayConfig() throws Exception { }
 5 
 6     public String getAppID() {
 7         return "wx4d4838ebec29b8** "; //你的appid
 8     }
 9 
10     public String getMchID() {
11         return "15508070**";  //你的商户号mch_id
12     }
13 
14     public String getKey() {
15         return "wHtQckdfiRBVF7ceGTcSWEEORt6C0D**";    //你的商户号秘钥 key
16     }
17 
18     public InputStream getCertStream() {
19         ByteArrayInputStream certBis = new ByteArrayInputStream(this.certData);
20         return certBis;
21     }
22 
23     public int getHttpConnectTimeoutMs() {
24         return 8000;
25     }
26 
27     public int getHttpReadTimeoutMs() {
28         return 10000;
29     }
30     public IWXPayDomain getWXPayDomain() {
31         // 这个方法需要这样实现, 否则无法正常初始化WXPay
32         IWXPayDomain iwxPayDomain = new IWXPayDomain() {
33 
34             public void report(String domain, long elapsedTimeMillis, Exception ex) {
35 
36             }
37 
38             public DomainInfo getDomain(WXPayConfig config) {
39                 return new IWXPayDomain.DomainInfo(WXPayConstants.DOMAIN_API, true);    //微信工具常量类有 "api.mch.weixin.qq.com";   wxpay.unifiedorder()  /pay/unifiedorder
40      }
41   };
42   return iwxPayDomain;
43
44 }
45
46 }

 找到 SDK 中的 WxPay修改里面的代码 

 1     public WXPay(final WXPayConfig config, final String notifyUrl, final boolean autoReport, final boolean useSandbox) throws Exception {
 2         this.config = config;
 3         this.notifyUrl = notifyUrl;
 4         this.autoReport = autoReport;
 5         this.useSandbox = useSandbox;
 6         if (useSandbox) {
 7             this.signType = SignType.MD5; // 沙箱环境
 8         }
 9         else {
10            // this.signType = SignType.HMACSHA256;  //注意:这点是个坑!   默认是HMACSHAS56加密 一定要修改成MD5  不然无论如何都会报  “微信签名失败” 的错误!
11             this.signType = SignType.MD5;
12         }
13         this.wxPayRequest = new WXPayRequest(config);
14     }

 

 

微信支付接口所需要的参数(前端): https://developers.weixin.qq.com/miniprogram/dev/api/open-api/payment/wx.requestPayment.html

可知 前端所需要的参数  我们直接反回给他们 就ok!

 

java微信支付代码

    //你自己需要定义一个方法
  public static void main(String[] args) throws Exception { //统一下单支付 HashMap<String, String> map = new HashMap<>(); IdWorker idWorker = new IdWorker();      //自定义订单号类 long out_trade_no = idWorker.nextId();    //获取订单号 SortedMap<Object, Object> parameters = new TreeMap<Object, Object>(); Map<String, String> data = new HashMap<>(); data.put("body", "微信支付"); //商品描述 data.put("total_fee", "1"); // 标价金额 单位:分 data.put("openid", "你传来的openid"); //用户标识 trade_type=JSAPI,此参数必传,用户在商户appid下的唯一标识 data.put("out_trade_no", out_trade_no + ""); //商户系统内部订单号 data.put("nonce_str",WxpayUtil.generateNonceStr()); //随机字符串,长度要求在32位以内。推荐随机数生成算法 data.put("spbill_create_ip", WeiXinHelper.localIp()); //支持IPV4和IPV6两种格式的IP地址。调用微信支付API的机器IP 自定获取ip data.put("notify_url", "http://www.weixin.qq.com/wxpay/pay.php"); // 没用到.通知地址:通知url必须为外网可访问的url,不能携带参数。 data.put("trade_type", "JSAPI");  //交易类型 data.put("sign_type", WXPayConstants.MD5); //签名类型//MyWxPayConfig 配置了一些默认信息 appid,商户号,商户秘钥,请求域名 .. MyWxPayConfig myWxPayConfig = new MyWxPayConfig(); WXPay wxpay = new WXPay(myWxPayConfig); Map<String, String> rMap = wxpay.unifiedOrder(data);  //生成一次签名 sign System.out.println(rMap);

     // 下面只是为了生成第二次签名 仅此而已 String return_code
= rMap.get("return_code");//返回状态码 String result_code = rMap.get("result_code");//结果状态码
     String nonce_str = rMap.get("nonce_str"); //随即字符串
     
Long s = System.currentTimeMillis() / 1000;  //获取时间戳除以千变字符串 String timeStamp = String.valueOf(s); if ("SUCCESS".equals(return_code) && return_code.equals(result_code)) {
        map.put(
"appId", “appid”);    //你的appid map.put("timeStamp", timeStamp);//这边要将返回的时间戳转化成字符串,不然小程序端调用wx.requestPayment方法会报签名错误 map.put("nonceStr", nonce_str); map.put("package", "prepay_id=" + rMap.get("prepay_id")); map.put("signType", "MD5"); System.out.println("二次签名参数 : " + map);  //需要生成二次签名 所用的参数 //再次签名sign,这个签名用于小程序端调用wx.requesetPayment方法 String sign = WXPayUtil.generateSignature(map, "key");  //你的商户号key map.put("paySign", sign); // 生成签名 重要 System.out.println("生成的签名paySign : " + sign);        // return map; //将map响应给前端 微信支付接口需要的参数 } }

 测试:打印结果

第一次签名后生成的数据 主要是   支付交易会话标识:prepay_id  

第二次签名后 再次组装数据 返回给前端的数据  wx.requestPayment  需要接收的数据

 

 容易遇到的错误 !   容易遇到的错误 !    容易遇到的错误 !

  1 商户号key 不要与 appid 的secret 弄混淆了

   2 SDK 工具类中 Wxpay 类中         this.signType = SignType.HMACSHA256;  HMACSHA256 改成 MD5

  3 第二次签名需要的五个参数一个不能少   appId,nonceStr,package,signType,timeStamp   。 注意 都是以 驼峰命名  不然也会报错

  

  成功!

   


鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
热门推荐
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap