❶ 求使用java怎樣配置微信jsSDK的使用方法 wx.config
wx.config({
debug: true, // 開啟調試模式,調用的所有api的返回值會在客戶端alert出來,若要查看傳入的參數,可以在pc端打開,參數信息會通過log打出,僅在pc端時才會列印。
appId: '', // 必填,公眾號的唯一標識
timestamp: , // 必填,生成簽名的時間戳
nonceStr: '', // 必填,生成簽名的隨機串
signature: '',// 必填,簽名,見附錄1
jsApiList: [] // 必填,需要使用的JS介面列表,所有JS介面列表見附錄2
});
❷ redis 怎麼使自增長主鍵調過一段繼續生成
資料庫自增什麼的麻煩死了,尤其是往後還需要考慮到分布式處理,然後偷了個懶,直接redis來搞起...
下面上代碼
先定義個主鍵生成策略介面,往後方便擴展
[java] view plain
/**
* 定義主鍵生成策略介面,以便修改擴展
* @author LeiYong
*
*/
public interface KeyGenerate {
/**
* 生成String類型主鍵
* @param em
* @return
*/
public String generateStringKey(KeyGenerateEnum em);
/**
* 生成long類型主鍵
* @param em
* @return
*/
public Long generateLongKey(KeyGenerateEnum em);
}
接下來提供一個替補隊員,基礎的主鍵生成方案,採用隨機數(Long)或UUID(String)方式生成
[java] view plain
/**
* 基礎主鍵生成策略,採用隨機數或UUID+隨機數
* @author LeiYong
*
*/
public class BaseKeyGenerate implements KeyGenerate{
@Override
public String generateStringKey(KeyGenerateEnum em) {
return StringUtil.getUUID()+StringUtil.getNonceStr(6);
}
@Override
public Long generateLongKey(KeyGenerateEnum em) {
return System.currentTimeMillis()*1000000+NumberUtil.random(6);
}
}
主角Redis的緩存主鍵類登場
[java] view plain
/**
* 採用Redis的方式產生自增長主鍵
* @author LeiYong
*
*/
public class RedisKeyGenerate implements KeyGenerate{
/**
* Logger for this class
*/
private final Logger logger = LoggerFactory.getLogger(RedisKeyGenerate.class);
private final String SEQUENCE_KEY = "SEQ";
@Autowired
@Qualifier("redisClientSupport")
private RedisClientSupport redisClientSupport;
@Override
public String generateStringKey(KeyGenerateEnum em) {
Long key = generateLongKey(em);
if (key!=null) {
return key.toString();
}
//為了防止redis掛了程序能夠繼續運行
return new BaseKeyGenerate().generateStringKey(em);
}
@Override
public Long generateLongKey(KeyGenerateEnum em) {
synchronized (em) {
try {
redisClientSupport.selectDB(SystemConfig.REDIS_INDEX_DB);
Long key = redisClientSupport.incrementHash(SEQUENCE_KEY, em.name(), 1l);
if (key != null) {
return key;
}
} catch (CacheAccessException e) {
logger.warn("generateStringKey(KeyGenerateEnum) - exception ignored", e); //$NON-NLS-1$
}
}
//為了防止redis掛了程序能夠繼續運行
return new BaseKeyGenerate().generateLongKey(em);
}
}
❸ 如何調用微信js
你好,調用微信js最簡單的辦法,就是在java代碼調用微信的Android SDK(具體用法看微信開發品台的示例,很回簡單), 在答js里使用jsb.reflection調用你的java方法。IOS也一樣。
❹ invalid signature,該怎麼辦
1.首先確認簽名演算法是正確的,到http://mp.weixin.qq.com/debug/cgi-bin/sandbox?t=jsapisign頁面工具進行校驗。也就是你自己後台生成簽名要和微信校驗演算法生成的簽名一致才可以。
2.還有一個筆者曾遇到的坑,看到微信SDK上說token和ticket需要在伺服器做緩存,有效期是7200秒,寫代碼的時候沒有仔細思索,導致把生成的簽名也給緩存了,用幾次就報invalid signature,後來仔細讀文檔之後,果斷改為每次打開頁面都重新生成簽名即可,token和ticket依然被緩存,但timestamp是變化的,因此每次簽名也是不同的。
3.參與簽名的欄位包括有效的 jsapi_ticket(獲取方式詳見微信JSSDK文檔), noncestr (隨機字元串,由開發者隨機生成),timestamp (由開發者生成的當前時間戳), url(當前網頁的URL,不包含#及其後面部分。
4.特別注意:你在利用參數生成簽名的時候,要對所有待簽名參數按照欄位名的 ASCII 碼從小到大排序(字典序)後,使用 URL 鍵值對的格式(即key1=value1&key2=value2…)拼接成字元串 string1。這里需要注意的是所有參數名均為小寫字元。
5.確認config中nonceStr(js中駝峰標准大寫S), timestamp與用以簽名中的對應noncestr, timestamp一致。
6.確認url是頁面完整的url(請在當前頁面alert(location.href.split('#')[0])確認),包括'http(s)://'部分,以及'?'後面的GET參數部分,但不包括'#'hash後面的部分。因為頁面一旦分享,微信客戶端會在你的鏈接末尾加入其它參數,如果不是動態獲取當前鏈接,將導致分享後的頁面簽名失敗。
❺ 微信支付後端篇
微信支付系列文章
微信支付-java後端實現
微信支付-vue 前端實現
java demo: 下載地址文章底部
技術棧
Spring boot
java
XML (微信在http協議中數據傳輸方案)
MD5 簽名
微信支付術語
openid (OpenID是公眾號一對一對應用戶身份的標識)
app_id (公眾號id,登錄微信公眾號–開發–基本配置中獲得;)
key (收款商戶後台進行配置,登錄微信商戶平台–賬戶中心–API安全-設置秘鑰,設置32位key值;)
mch_id (收款商家商戶號;)
certPath (API證書, 登錄微信商戶平台–賬戶中心-API安全-下載證書)
後端流程
服務端需要的核心操作, 總共分為以下幾步:
統一下單
前端調起微信支付必要參數 (需加密)
訂單結果主動通知 (回調介面)
查詢訂單結果
結束訂單支付介面(關閉訂單,支付訂單關閉)
代碼
微信總共支持多種語言的sdk, 在官網可以下載例子, java程序也可以引入微信支付的sdk包, 但是github上的sdk已經很久沒有更新了, 最好的選擇, 也是我的選擇, 在官網上下載sdk項目, 將其中所有java類到自己的項目中.
官網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();
}
@Override
public String getAppID() {
return app_id;
}
@Override
public String getMchID() {
return wx_pay_mch_id;
}
@Override
public String getKey() {
return wx_pay_key;
}
@Override
public InputStream getCertStream() {
return new ByteArrayInputStream(this.certData);
}
@Override
public IWXPayDomain getWXPayDomain() { // 這個方法需要這樣實現, 否則無法正常初始化WXPay
IWXPayDomain iwxPayDomain = new IWXPayDomain() {
@Override
public void report(String domain, long elapsedTimeMillis, Exception ex) {
}
@Override
public DomainInfo getDomain(WXPayConfig config) {
return new IWXPayDomain.DomainInfo(WXPayConstants.DOMAIN_API, true);
}
};
return iwxPayDomain;
}
}
發起統一下單 AND 前端調起微信支付必要參數
// 發起微信支付
WXPay wxpay = null;
Map result = new HashMap>();
try {
// ******************************************
//
// 統一下單
//
// ******************************************
wxpay = new WXPay(iWxPayConfig); // *** 注入自己實現的微信配置類, 創建WXPay核心類, WXPay 包括統一下單介面
Map data = new HashMap ();
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"); // 下單ip
data.put("openid", openId); // 微信公眾號統一標示openid
data.put("notify_url", "http://wxlj.oopmind.com/payCallback"); // 訂單結果通知, 微信主動回調此介面
data.put("trade_type", "JSAPI"); // 固定填寫
logger.info("發起微信支付下單介面, request={}", data);
Map 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 wxPayMap = new HashMap ();
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轉換成String
BufferedReader reader = new BufferedReader(new InputStreamReader(is));
StringBuilder sb = new StringBuilder();
String line = null;
try {
while ((line = reader.readLine()) != null) {
sb.append(line + " ");
}
} 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 = " ";
return result;
}
}
@Override
public String payBack(String notifyData) {
logger.info("payBack() start, notifyData={}", notifyData);
String xmlBack="";
Map notifyMap = null;
try {
WXPay wxpay = new WXPay(iWxPayConfig);
notifyMap = WXPayUtil.xmlToMap(notifyData); // 轉換成map
if (wxpay.(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 = " ";
return xmlBack;
}
// 業務邏輯處理 ****************************
logger.info("微信支付回調成功訂單號: {}", notifyMap);
xmlBack = " ";
return xmlBack;
} else {
logger.error("微信支付回調通知簽名錯誤");
xmlBack = " ";
return xmlBack;
}
} catch (Exception e) {
logger.error("微信支付回調通知失敗",e);
xmlBack = " ";
}
return xmlBack;
}
統一下單的簽名和後續前端拉取微信支付的簽名需要統一, 也就是都採用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);
}
結束語
做完以後, 微信支付的後端邏輯還是很清晰的, 但是在開發過程中很煎熬, 不清楚每個專業術語在微信哪裡配置, 加密方式亂的很
❻ 怎麼用java調用微信支付介面
java調用微信支付介面方法:x0d=newRequestHandler(super.getRequest(),super.getResponse());x0dx0ax0dx0a//獲取token//兩小時內有效,兩小時後重新獲取x0dx0ax0dx0aToken=requestHandler.GetToken();x0dx0ax0dx0a//更新token到應用中x0dx0ax0dx0arequestHandler.getTokenReal();x0dx0ax0dx0aSystem.out.println("微信支付獲取token=======================:"+Token);x0dx0ax0dx0ax0dx0ax0dx0a//requestHandler初始化x0dx0ax0dx0arequestHandler.init();x0dx0ax0dx0arequestHandler.init(appid,appsecret,appkey,partnerkey,key);x0dx0ax0dx0ax0dx0ax0dx0a//--------------------------------本地系統生成訂單-------------------------------------x0dx0ax0dx0a//設置package訂單參數x0dx0ax0dx0aSortedMap