java遇到微信小程序 "支付驗(yàn)證簽名失敗" 問(wèn)題解決
最近在做一個(gè)微信小程序項(xiàng)目做到微信支付的時(shí)候遇到的一些問(wèn)題!
詳細(xì)步驟:
開(kāi)發(fā)前準(zhǔn)備(必須)
小程序標(biāo)識(shí)(appid):wx4d4838ebec29b8**
商戶(hù)號(hào)(mch_id):15508070**
商戶(hù)密鑰(key) :wHtQckdfiRBVF7ceGTcSWEEORt6C0D**
我們用微信官方提供的SDK開(kāi)發(fā) :https://pay.weixin.qq.com/wiki/doc/api/native.php?chapter=11_1

下載 SDK完成后 : 
開(kāi)始寫(xiě)我們的程
進(jìn)入微信支付開(kāi)發(fā)文檔 :https://pay.weixin.qq.com/wiki/doc/api/index.html
選擇 小程序支付

選擇 API列表 統(tǒng)一下單可以看到微信接口鏈接和 請(qǐng)求參數(shù) , 你需要看下每個(gè)參數(shù)什么意思,接下來(lái)就需要知道怎么操作這些參數(shù)就可以了 ok

一 首先 把剛下載的 微信提供的 SDK 拷貝到你的項(xiàng)目里 自定義一個(gè)類(lèi)繼承里面的一個(gè)WXPayConfig 抽象類(lèi)
public class MyWxPayConfig extends WXPayConfig {
private byte[] certData;
public MyWxPayConfig() throws Exception { }
public String getAppID() {
return "wx4d4838ebec29b8** "; //你的appid
}
public String getMchID() {
return "15508070**"; //你的商戶(hù)號(hào)mch_id
}
public String getKey() {
return "wHtQckdfiRBVF7ceGTcSWEEORt6C0D**"; //你的商戶(hù)號(hào)秘鑰 key
}
public InputStream getCertStream() {
ByteArrayInputStream certBis = new ByteArrayInputStream(this.certData);
return certBis;
}
public int getHttpConnectTimeoutMs() {
return 8000;
}
public int getHttpReadTimeoutMs() {
return 10000;
}
public IWXPayDomain getWXPayDomain() {
// 這個(gè)方法需要這樣實(shí)現(xiàn), 否則無(wú)法正常初始化WXPay
IWXPayDomain iwxPayDomain = new IWXPayDomain() {
public void report(String domain, long elapsedTimeMillis, Exception ex) {
}
public DomainInfo getDomain(WXPayConfig config) {
return new IWXPayDomain.DomainInfo(WXPayConstants.DOMAIN_API, true); //微信工具常量類(lèi)有 "api.mch.weixin.qq.com"; wxpay.unifiedorder() /pay/unifiedorder
}
};
return iwxPayDomain;
}
}
找到 SDK 中的 WxPay 類(lèi) 修改里面的代碼
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; // 沙箱環(huán)境
}
else {
// this.signType = SignType.HMACSHA256; //注意:這點(diǎn)是個(gè)坑! 默認(rèn)是HMACSHAS56加密 一定要修改成MD5 不然無(wú)論如何都會(huì)報(bào) “微信簽名失敗” 的錯(cuò)誤!
this.signType = SignType.MD5;
}
this.wxPayRequest = new WXPayRequest(config);
}
微信支付接口所需要的參數(shù)(前端):https://developers.weixin.qq.com/miniprogram/dev/api/open-api/payment/wx.requestPayment.html

可知 前端所需要的參數(shù) 我們直接反回給他們 就ok!
java微信支付代碼
//你自己需要定義一個(gè)方法
public static void main(String[] args) throws Exception {
//統(tǒng)一下單支付
HashMap<String, String> map = new HashMap<>();
IdWorker idWorker = new IdWorker(); //自定義訂單號(hào)類(lèi)
long out_trade_no = idWorker.nextId(); //獲取訂單號(hào)
SortedMap<Object, Object> parameters = new TreeMap<Object, Object>();
Map<String, String> data = new HashMap<>();
data.put("body", "微信支付"); //商品描述
data.put("total_fee", "1"); // 標(biāo)價(jià)金額 單位:分
data.put("openid", "你傳來(lái)的openid"); //用戶(hù)標(biāo)識(shí) trade_type=JSAPI,此參數(shù)必傳,用戶(hù)在商戶(hù)appid下的唯一標(biāo)識(shí)
data.put("out_trade_no", out_trade_no + ""); //商戶(hù)系統(tǒng)內(nèi)部訂單號(hào)
data.put("nonce_str",WxpayUtil.generateNonceStr()); //隨機(jī)字符串,長(zhǎng)度要求在32位以?xún)?nèi)。推薦隨機(jī)數(shù)生成算法
data.put("spbill_create_ip", WeiXinHelper.localIp()); //支持IPV4和IPV6兩種格式的IP地址。調(diào)用微信支付API的機(jī)器IP 自定獲取ip
data.put("notify_url", "http://www.weixin.qq.com/wxpay/pay.php"); // 沒(méi)用到.通知地址:通知url必須為外網(wǎng)可訪(fǎng)問(wèn)的url,不能攜帶參數(shù)。
data.put("trade_type", "JSAPI"); //交易類(lèi)型
data.put("sign_type", WXPayConstants.MD5); //簽名類(lèi)型//MyWxPayConfig 配置了一些默認(rèn)信息 appid,商戶(hù)號(hào),商戶(hù)秘鑰,請(qǐng)求域名 ..
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");//返回狀態(tài)碼
String result_code = rMap.get("result_code");//結(jié)果狀態(tài)碼
String nonce_str = rMap.get("nonce_str"); //隨即字符串
Long s = System.currentTimeMillis() / 1000; //獲取時(shí)間戳除以千變字符串
String timeStamp = String.valueOf(s);
if ("SUCCESS".equals(return_code) && return_code.equals(result_code)) {
map.put("appId", “appid”); //你的appid
map.put("timeStamp", timeStamp);//這邊要將返回的時(shí)間戳轉(zhuǎn)化成字符串,不然小程序端調(diào)用wx.requestPayment方法會(huì)報(bào)簽名錯(cuò)誤
map.put("nonceStr", nonce_str);
map.put("package", "prepay_id=" + rMap.get("prepay_id"));
map.put("signType", "MD5");
System.out.println("二次簽名參數(shù) : " + map); //需要生成二次簽名 所用的參數(shù)
//再次簽名sign,這個(gè)簽名用于小程序端調(diào)用wx.requesetPayment方法
String sign = WXPayUtil.generateSignature(map, "key"); //你的商戶(hù)號(hào)key
map.put("paySign", sign); // 生成簽名 重要
System.out.println("生成的簽名paySign : " + sign);
// return map; //將map響應(yīng)給前端 微信支付接口需要的參數(shù)
}
}
測(cè)試:打印結(jié)果
第一次簽名后生成的數(shù)據(jù) 主要是 支付交易會(huì)話(huà)標(biāo)識(shí):prepay_id

第二次簽名后 再次組裝數(shù)據(jù) 返回給前端的數(shù)據(jù) wx.requestPayment 需要接收的數(shù)據(jù)

容易遇到的錯(cuò)誤 ! 容易遇到的錯(cuò)誤 ! 容易遇到的錯(cuò)誤 !
1 商戶(hù)號(hào)key 不要與 appid 的secret 弄混淆了
2 SDK 工具類(lèi)中 Wxpay 類(lèi)中 this.signType = SignType.HMACSHA256; HMACSHA256 改成 MD5
3 第二次簽名需要的五個(gè)參數(shù)一個(gè)不能少 appId,nonceStr,package,signType,timeStamp 。 注意 都是以駝峰命名 不然也會(huì)報(bào)錯(cuò)
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
記錄一次開(kāi)發(fā)微信網(wǎng)頁(yè)分享的步驟
這篇文章主要介紹了記錄一次開(kāi)發(fā)微信網(wǎng)頁(yè)分享的步驟,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-05-05
兩種常用的javascript數(shù)組去重方法思路及代碼
第一種是常規(guī)的方法:建一個(gè)新的數(shù)組存放結(jié)果,for循環(huán)中每次從原數(shù)組中取出一個(gè)元素,用indexOf查找新數(shù)組中是否有該元素,至于第二種詳細(xì)的看下本文哦2013-03-03
laydate如何根據(jù)開(kāi)始時(shí)間或者結(jié)束時(shí)間限制范圍
這篇文章主要為大家詳細(xì)介紹了laydate根據(jù)開(kāi)始時(shí)間或者結(jié)束時(shí)間限制范圍的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-11-11
layui 給數(shù)據(jù)表格加序號(hào)的方法
今天小編就為大家分享一篇layui 給數(shù)據(jù)表格加序號(hào)的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-08-08
JS 屏蔽鍵盤(pán)不可用與鼠標(biāo)右鍵不可用的方法
這篇文章主要介紹了JS 屏蔽鍵盤(pán)不可用與鼠標(biāo)右鍵不可用的方法,有需要的朋友可以參考一下2013-11-11
JS循環(huán)發(fā)送請(qǐng)求時(shí)控制請(qǐng)求并發(fā)數(shù)實(shí)例
這篇文章主要介紹了JS循環(huán)發(fā)送請(qǐng)求時(shí)控制請(qǐng)求并發(fā)數(shù)實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12

