小程序開發調用微信支付以及微信回調地址配置
首先觀看微信提供的文檔
https://pay.weixin.qq.com/wiki/doc/api/wxa/wxa_api.php?chapter=7_3&index=1
清楚調用微信支付必須傳遞的參數
因為微信提供瞭小程序喚起微信支付的方法,後端隻需要傳遞對應的參數給前端即可
首先在程序中配置申請的固定參數
wx.open.app_id=用戶的appid wx.open.app_secret=這是做登陸用的 weixin.pay.partner=商戶號 wexxin.pay.partenerkey=商戶號秘鑰
編寫工具類實現對固定值的讀取
@Component //@PropertySource("classpath:application.properties") public class ConstantPropertiesUtil implements InitializingBean { //讀取配置文件並賦值 @Value("${wx.open.app_id}") private String appId; @Value("${wx.open.app_secret}") private String appSecret; @Value("{weixin.pay.partner}") private String partner; @Value("{wexxin.pay.partenerkey}") private String partenerkey; public static String WX_OPEN_APP_ID; public static String WX_OPEN_APP_SECRET; public static String PARTNER; public static String PARTNERKET; @Override public void afterPropertiesSet() throws Exception { WX_OPEN_APP_ID = appId; WX_OPEN_APP_SECRET = appSecret; PARTNER = partner; PARTNERKET = partenerkey; } }
當用戶點擊購買會生成訂單,這裡代碼省略
點擊登陸時調用後端傳給前端需要的值
對應微信文檔https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1
可以看到,除瞭一些固定值,需要我們自己處理的有
簽名:根據文檔可以發現簽名是有一定要求的
簡單來說就將其他傳入固定值字段進行排序拼接,在根據商傢號的key進行加密處理。
支付接口
@Autowired private WXService wxService; @GetMapping("pay") public R creatNative(Integer orderid){ try { Map map = wxService.payment(orderid); return R.ok().data(map); } catch (UnsupportedEncodingException e) { return R.error().message("支付失敗"); } }
編寫service邏輯,根據文檔進行傳值
@Service public class WXServiceImpl implements WXService { @Autowired private OrderService orderService; @Override public Map payment(Integer orderid) throws UnsupportedEncodingException { //封裝傳遞微信地址參數 Map paramMap = new HashMap(); paramMap.put("appid", ConstantPropertiesUtil.WX_OPEN_APP_ID); //公眾號id paramMap.put("mch_id", ConstantPropertiesUtil.PARTNER); //商戶號 paramMap.put("nonce_str", WXPayUtil.generateNonceStr()); //隨機字符串,調用工具類 paramMap.put("out_trade_no", orderid); //訂單流水號 Order order = orderService.getById(orderid); paramMap.put("total_fee", order.getPayment()); //金額 paramMap.put("spbill_create_ip", "127.0.0.1"); //終端ip paramMap.put("notify_url", "http://XXXXX/weixin/callBack");//回調地址 paramMap.put("body",order.getProductname()); //商品名稱 paramMap.put("timeStamp", WXUtil.getCurrentTimestamp()+"");//獲取當前時間戳,單位秒 String sign = WXUtil.genSignature(ConstantPropertiesUtil.PARTNERKET,paramMap); //sing paramMap.put("sign", sign); //簽名 return paramMap; } }
簽名工具類,以及時間戳方法
public class WXUtil { public static String genSignature(String secretKey, Map<String, String> params) throws UnsupportedEncodingException { if (secretKey == null || params == null || params.size() == 0) { return ""; } // 1. 參數名按照ASCII碼表升序排序 String[] keys = params.keySet().toArray(new String[0]); Arrays.sort(keys); // 2. 按照排序拼接參數名與參數值 StringBuffer paramBuffer = new StringBuffer(); for (String key : keys) { paramBuffer.append("&"+key).append(params.get(key) == null ? "" : "="+params.get(key)); } // 3. 將secretKey拼接到最後 paramBuffer=paramBuffer.append("&key="+secretKey); String pa =paramBuffer.substring(1); // 4. MD5是128位長度的摘要算法,用16進制表示,一個十六進制的字符能表示4個位,所以簽名後的字符串長度固定為32個十六進制字符。 return DigestUtils.md5Hex(pa.getBytes("UTF-8")).toUpperCase(); } /** * 獲取當前時間戳,單位秒 * @return */ public static long getCurrentTimestamp() { return System.currentTimeMillis()/1000; } /** * 獲取當前時間戳,單位毫秒 * @return */ public static long getCurrentTimestampMs() { return System.currentTimeMillis(); } }
此時即可完成支付,微信支付後,微信會給我們回調地址進行發送信息,由此我們可以判斷支付狀態以及獲取微信支付返回的參數
回調接口
//回調接口 @RequestMapping("callBack") public String callBack(HttpServletRequest request, HttpServletResponse response) throws Exception{ System.out.println("接口已被調用"); ServletInputStream inputStream = request.getInputStream(); String notifyXml = StreamUtils.inputStream2String(inputStream, "utf-8"); System.out.println(notifyXml); // 解析返回結果 Map<String, String> notifyMap = WXPayUtil.xmlToMap(notifyXml); // 判斷支付是否成功 if ("SUCCESS".equals(notifyMap.get("result_code"))) { //編寫自己的實現邏輯 // 支付成功:給微信發送我已接收通知的響應 // 創建響應對象 Map<String, String> returnMap = new HashMap<>(); returnMap.put("return_code", "SUCCESS"); returnMap.put("return_msg", "OK"); String returnXml = WXPayUtil.mapToXml(returnMap); response.setContentType("text/xml"); System.out.println("支付成功"); return returnXml; } } // 創建響應對象:微信接收到校驗失敗的結果後,會反復的調用當前回調函數 Map<String, String> returnMap = new HashMap<>(); returnMap.put("return_code", "FAIL"); returnMap.put("return_msg", ""); String returnXml = WXPayUtil.mapToXml(returnMap); response.setContentType("text/xml"); System.out.println("校驗失敗"); return returnXml; }
接收輸入流轉換工具類
public class StreamUtils { private static int _buffer_size = 1024; /** * InputStream流轉換成String字符串 * @param inStream InputStream流 * @param encoding 編碼格式 * @return String字符串 */ public static String inputStream2String(InputStream inStream, String encoding){ String result = null; ByteArrayOutputStream outStream = null; try { if(inStream != null){ outStream = new ByteArrayOutputStream(); byte[] tempBytes = new byte[_buffer_size]; int count = -1; while((count = inStream.read(tempBytes, 0, _buffer_size)) != -1){ outStream.write(tempBytes, 0, count); } tempBytes = null; outStream.flush(); result = new String(outStream.toByteArray(), encoding); outStream.close(); } } catch (Exception e) { result = null; } finally { try { if(inStream != null) { inStream.close(); inStream = null; } if(outStream != null) { outStream.close(); outStream = null; } } catch (IOException e) { e.printStackTrace(); } } return result; } }
到此這篇關於小程序開發調用微信支付以及微信回調地址配置的文章就介紹到這瞭,更多相關小程序 調用微信支付以及微信回調內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- JSch教程使用sftp協議實現服務器文件載操作
- Java 使用Socket正確讀取數據姿勢
- 重寫equals的同時為何要重寫hashCode?
- 普通類註入不進spring bean的解決方法
- Java中生成微信小程序太陽碼的實現方案