热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

用H5调用支付微信公众号支付的解析-

这篇文章主要为大家详细介绍了微信公众号支付H5调用支付,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
这篇文章主要为大家详细介绍了微信公众号支付H5调用支付,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

最近项目需要微信支付,然后看了下微信公众号支付,虽然不难,但是细节还是需要注意的,用了大半天时间写了个demo,并且完整的测试了一下支付流程,下面分享一下微信公众号支付的经验。

一、配置公众号微信支付

需要我们配置微信公众号支付地址和测试白名单。

在微信中发送如下链接

https://open.weixin.qq.com/connect/oauth2/authorize?appid=wx520c15f417810387&redirect_uri=要跳转的下订单的url&response_type=code&scope=snsapi_base&state=123#wechat_redirect

3.2 后台支付

代码如下,包含预处理订单,支付订单等接口。

package org.andy.controller; 
 
import java.io.ByteArrayInputStream; 
import java.io.IOException; 
import java.io.InputStream; 
import java.io.UnsupportedEncodingException; 
import java.util.Date; 
import java.util.HashMap; 
import java.util.Iterator; 
import java.util.Map; 
import java.util.Map.Entry; 
import java.util.Random; 
 
import javax.servlet.ServletInputStream; 
import javax.servlet.http.HttpServletRequest; 
import javax.servlet.http.HttpServletResponse; 
 
import org.apache.commons.codec.digest.DigestUtils; 
import org.springframework.stereotype.Controller; 
import org.springframework.web.bind.annotation.RequestMapping; 
import org.xmlpull.v1.XmlPullParser; 
import org.xmlpull.v1.XmlPullParserException; 
import org.xmlpull.v1.XmlPullParserFactory; 
 
import com.fasterxml.jackson.databind.JsonNode; 
import com.gson.oauth.Oauth; 
import com.gson.oauth.Pay; 
import com.gson.util.HttpKit; 
import com.gson.util.Tools; 
import org.andy.util.DatetimeUtil; 
import org.andy.util.JsonUtil; 
import org.andy.util.SessionUtil; 
import org.andy.util.WebUtil; 
 
@Controller 
@RequestMapping("/pay") 
public class WXPayController { 
 
 @RequestMapping(value = "wxprepay") 
 public void jspay(HttpServletRequest request, HttpServletResponse response, String callback) throws Exception { 
 // 获取openid 
 String openId = SessionUtil.getAtt(request, "openId"); 
 if (openId == null) { 
 openId = getUserOpenId(request); 
 } 
 
 String appid = "wx16691fcb0523c1a4"; 
 String partnerid = "22223670"; 
 String paternerKey = "fjfjfjfjf1234567FFFFFFFFF1234567"; 
 
 String out_trade_no = getTradeNo(); 
 Map paraMap = new HashMap(); 
 paraMap.put("appid", appid); 
 paraMap.put("attach", "测试支付"); 
 paraMap.put("body", "测试购买Beacon支付"); 
 paraMap.put("mch_id", partnerid); 
 paraMap.put("nonce_str", create_nonce_str()); 
 paraMap.put("openid", openId); 
 paraMap.put("out_trade_no", out_trade_no); 
 paraMap.put("spbill_create_ip", getAddrIp(request)); 
 paraMap.put("total_fee", "1"); 
 paraMap.put("trade_type", "JSAPI"); 
 paraMap.put("notify_url", "http://www.xxx.co/wxpay/pay/appPay_notify.shtml"); 
 String sign = getSign(paraMap, paternerKey); 
 paraMap.put("sign", sign); 
 
 // 统一下单 https://api.mch.weixin.qq.com/pay/unifiedorder 
 String url = "https://api.mch.weixin.qq.com/pay/unifiedorder"; 
 
 String xml = ArrayToXml(paraMap, false); 
 
 String xmlStr = HttpKit.post(url, xml); 
 
 // 预付商品id 
 String prepay_id = ""; 
 
 if (xmlStr.indexOf("SUCCESS") != -1) { 
 Map map = doXMLParse(xmlStr); 
 prepay_id = (String) map.get("prepay_id"); 
 } 
 
 Map payMap = new HashMap(); 
 payMap.put("appId", appid); 
 payMap.put("timeStamp", create_timestamp()); 
 payMap.put("nonceStr", create_nonce_str()); 
 payMap.put("signType", "MD5"); 
 payMap.put("package", "prepay_id=" + prepay_id); 
 String paySign = getSign(payMap, paternerKey); 
 
 payMap.put("pg", prepay_id); 
 payMap.put("paySign", paySign); 
 
 
 WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(payMap)).toString())); 
 } 
 
 @RequestMapping(value = "appPay") 
 public void appPay(HttpServletRequest request, HttpServletResponse response, String body, String detail, String total_fee, String spbill_create_ip, 
 String notify_url, String trade_type, String callback) throws Exception { 
 
 String appid = "wx16691fcb0523c1a4"; 
 String partnerid = "22223670"; 
 String paternerKey = "fjfjfjfjf1234567FFFFFFFFF1234567"; 
 
 String out_trade_no = getTradeNo(); 
 Map paraMap = new HashMap(); 
 paraMap.put("appid", appid); 
 paraMap.put("body", body); 
 paraMap.put("mch_id", partnerid); 
 paraMap.put("nonce_str", create_nonce_str()); 
 paraMap.put("out_trade_no", out_trade_no); 
 paraMap.put("spbill_create_ip", spbill_create_ip); 
 paraMap.put("total_fee", total_fee); 
 paraMap.put("trade_type", trade_type); 
 paraMap.put("notify_url", notify_url); 
 String sign = getSign(paraMap, paternerKey); 
 paraMap.put("sign", sign); 
 
 // 统一下单 https://api.mch.weixin.qq.com/pay/unifiedorder 
 String url = "https://api.mch.weixin.qq.com/pay/unifiedorder"; 
 
 String xml = ArrayToXml(paraMap, false); 
 
 String xmlStr = HttpKit.post(url, xml); 
 
 // 预付商品id 
 String prepay_id = ""; 
 
 Map map = doXMLParse(xmlStr); 
 if (xmlStr.indexOf("SUCCESS") != -1) { 
 prepay_id = (String) map.get("prepay_id"); 
 } 
 
 String result_code = map.get("result_code"); 
 String err_code_des = map.get("err_code_des"); 
 Map payMap = new HashMap(); 
 payMap.put("appid", appid); 
 payMap.put("partnerid", partnerid); 
 payMap.put("prepayid", prepay_id); 
 payMap.put("package", "Sign=WXPay"); 
 payMap.put("noncestr", create_nonce_str()); 
 payMap.put("timestamp", create_timestamp()); 
 String paySign = getSign(payMap, paternerKey); 
 
 payMap.put("sign", paySign); 
 payMap.put("result_code", result_code); 
 payMap.put("err_code_des", err_code_des); 
 
 
 WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(payMap)).toString())); 
 } 
 
 @RequestMapping("/appPay_notify") 
 public void appPayNotify(HttpServletRequest request, HttpServletResponse response) throws Exception{ 
 //String xml = "1"; 
 response.setCharacterEncoding("UTF-8"); 
 response.setContentType("text/xml"); 
 ServletInputStream in = request.getInputStream(); 
 String xmlMsg = Tools.inputStream2String(in); 
 
 Map map = doXMLParse(xmlMsg); 
 String return_code = map.get("return_code"); 
 String return_msg = map.get("return_msg"); 
 
 map = new HashMap(); 
 map.put("return_code", return_code); 
 map.put("return_msg", return_msg); 
 
 //响应xml 
 String resXml = ArrayToXml(map, true); 
 response.getWriter().write(resXml); 
 } 
 
 @RequestMapping("/orderquery.do") 
 public void orderquery(HttpServletRequest request, HttpServletResponse response, String transaction_id, String out_trade_no, String callback) throws Exception{ 
 
 String url = "https://api.mch.weixin.qq.com/pay/orderquery"; 
 
 String appid = "wx16691fcb0523c1a4"; 
 String partnerid = "22223670"; 
 String paternerKey = "fjfjfjfjf1234567FFFFFFFFF1234567"; 
 
 Map map = new HashMap(); 
 map.put("appid", appid); 
 map.put("mch_id", partnerid); 
 if(transaction_id != null && !transaction_id.equals("")){ 
 map.put("transaction_id", transaction_id); 
 }else { 
 map.put("out_trade_no", out_trade_no); 
 } 
 map.put("nonce_str", create_nonce_str()); 
 String paySign = getSign(map, paternerKey); 
 map.put("sign", paySign); 
 
 String xml = ArrayToXml(map, false); 
 String xmlStr = HttpKit.post(url, xml); 
 
 Map orderMap = doXMLParse(xmlStr); 
 
 
 WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(orderMap)).toString())); 
 } 
 
 /** 
 * map转成xml 
 * 
 * @param arr 
 * @return 
 */ 
 public String ArrayToXml(Map parm, boolean isAddCDATA) { 
 StringBuffer strbuff = new StringBuffer(); 
 if (parm != null ) { 
 for (Entry entry : parm.entrySet()) { 
 strbuff.append("<").append(entry.getKey()).append(">"); 
 if (isAddCDATA) { 
  strbuff.append(); 
 }else { 
  strbuff.append(entry.getValue()); 
 } 
 strbuff.append("<").append(entry.getKey()).append(">"); 
 } 
 } 
 return strbuff.append().toString(); 
 } 
 
 // 获取openId 
 private String getUserOpenId(HttpServletRequest request) throws Exception { 
 String code = request.getParameter("code"); 
 if (code == null) { 
 String openId = request.getParameter("openId"); 
 return openId; 
 } 
 Oauth o = new Oauth(); 
 String token = o.getToken(code); 
 JsonNode node = JsonUtil.StringToJsonNode(token); 
 String openId = node.get("openid").asText(); 
 return openId; 
 } 
 
 private String create_nonce_str() { 
 String chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"; 
 String res = ""; 
 for (int i = 0; i <16; i++) { 
 Random rd = new Random(); 
 res += chars.charAt(rd.nextInt(chars.length() - 1)); 
 } 
 return res; 
 } 
 
 private String getAddrIp(HttpServletRequest request){ 
 return request.getRemoteAddr(); 
 } 
 
 private String create_timestamp() { 
 return Long.toString(System.currentTimeMillis() / 1000); 
 } 
 
 private String getTradeNo(){ 
 String timestamp = DatetimeUtil.formatDate(new Date(), DatetimeUtil.DATETIME_PATTERN); 
 return "HZNO" + timestamp; 
 } 
 
 private String getSign(Map params, String paternerKey ) 
 throws UnsupportedEncodingException { 
 String string1 = Pay.createSign(params, false); 
 String stringSignTemp = string1 + "&key=" + paternerKey; 
 String signValue = DigestUtils.md5Hex(stringSignTemp).toUpperCase(); 
 return signValue; 
 } 
 
 private Map doXMLParse(String xml) 
 throws XmlPullParserException, IOException { 
 
 InputStream inputStream = new ByteArrayInputStream(xml.getBytes()); 
 
 Map map = null; 
 
 XmlPullParser pullParser = XmlPullParserFactory.newInstance() 
 .newPullParser(); 
 
 pullParser.setInput(inputStream, "UTF-8"); // 为xml设置要解析的xml数据 
 
 int eventType = pullParser.getEventType(); 
 
 while (eventType != XmlPullParser.END_DOCUMENT) { 
 switch (eventType) { 
 case XmlPullParser.START_DOCUMENT: 
 map = new HashMap(); 
 break; 
 
 case XmlPullParser.START_TAG: 
 String key = pullParser.getName(); 
 if (key.equals("xml")) 
  break; 
 
 String value = pullParser.nextText(); 
 map.put(key, value); 
 
 break; 
 
 case XmlPullParser.END_TAG: 
 break; 
 
 } 
 
 eventType = pullParser.next(); 
 
 } 
 
 return map; 
 } 
 
}

wxprepay.shtm接口是预处理订单接口向微信服务器下订单。
appPay.shtml接口是支付接口。
appPay_notify.shtml接口是微信支付后异步通知结果接口。
orderquery.shtml接口是订单查询接口

3.3、涉及到的工具类
SessionUtil.java工具类

package org.andy.util; 
 
import javax.servlet.http.HttpServletRequest; 
 
 
public class SessionUtil { 
 public static void addAtt(HttpServletRequest request, String key, Object value){ 
 request.getSession().setAttribute(key, value); 
 } 
 
 public static void removeAtt(HttpServletRequest request, String key){ 
 request.getSession().removeAttribute(key); 
 } 
 
 public static String getAtt(HttpServletRequest request, String key){ 
 return (String)request.getSession().getAttribute(key); 
 } 
 
 public static Object getAttObj(HttpServletRequest request, String key){ 
 return request.getSession().getAttribute(key); 
 } 
 
 public static String optAtt(HttpServletRequest request, String key, String value){ 
 String r = (String)request.getSession().getAttribute(key); 
 if (r == null){ 
 r = value; 
 } 
 return r; 
 } 
 
}

HttpKit 网络请求工具类

/** 
 * https 请求 微信为https的请求 
 * 
 * @author andy 
 * @date 2015-10-9 下午2:40:19 
 */ 
public class HttpKit { 
 private static final String DEFAULT_CHARSET = "UTF-8"; 
 /** 
 * @return 返回类型: 
 * @throws IOException 
 * @throws UnsupportedEncodingException 
 * @throws NoSuchProviderException 
 * @throws NoSuchAlgorithmException 
 * @throws KeyManagementException 
 * @description 功能描述: get 请求 
 */ 
 public static String get(String url, Map params, Map headers) throws IOException, ExecutionException, InterruptedException { 
 AsyncHttpClient http = new AsyncHttpClient(); 
 AsyncHttpClient.BoundRequestBuilder builder = http.prepareGet(url); 
 builder.setBodyEncoding(DEFAULT_CHARSET); 
 if (params != null && !params.isEmpty()) { 
 Set keys = params.keySet(); 
 for (String key : keys) { 
 builder.addQueryParameter(key, params.get(key)); 
 } 
 } 
 
 if (headers != null && !headers.isEmpty()) { 
 Set keys = headers.keySet(); 
 for (String key : keys) { 
 builder.addHeader(key, params.get(key)); 
 } 
 } 
 Future f = builder.execute(); 
 String body = f.get().getResponseBody(DEFAULT_CHARSET); 
 http.close(); 
 return body; 
 } 
 
 /** 
 * @return 返回类型: 
 * @throws IOException 
 * @throws UnsupportedEncodingException 
 * @throws NoSuchProviderException 
 * @throws NoSuchAlgorithmException 
 * @throws KeyManagementException 
 * @description 功能描述: get 请求 
 */ 
 public static String get(String url) throws KeyManagementException, NoSuchAlgorithmException, NoSuchProviderException, UnsupportedEncodingException, IOException, ExecutionException, InterruptedException { 
 return get(url, null); 
 } 
 
 /** 
 * @return 返回类型: 
 * @throws IOException 
 * @throws NoSuchProviderException 
 * @throws NoSuchAlgorithmException 
 * @throws KeyManagementException 
 * @throws UnsupportedEncodingException 
 * @description 功能描述: get 请求 
 */ 
 public static String get(String url, Map params) throws KeyManagementException, NoSuchAlgorithmException, NoSuchProviderException, UnsupportedEncodingException, IOException, ExecutionException, InterruptedException { 
 return get(url, params, null); 
 } 
 
 /** 
 * @return 返回类型: 
 * @throws IOException 
 * @throws NoSuchProviderException 
 * @throws NoSuchAlgorithmException 
 * @throws KeyManagementException 
 * @description 功能描述: POST 请求 
 */ 
 public static String post(String url, Map params) throws IOException, ExecutionException, InterruptedException { 
 AsyncHttpClient http = new AsyncHttpClient(); 
 AsyncHttpClient.BoundRequestBuilder builder = http.preparePost(url); 
 builder.setBodyEncoding(DEFAULT_CHARSET); 
 if (params != null && !params.isEmpty()) { 
 Set keys = params.keySet(); 
 for (String key : keys) { 
 builder.addParameter(key, params.get(key)); 
 } 
 } 
 Future f = builder.execute(); 
 String body = f.get().getResponseBody(DEFAULT_CHARSET); 
 http.close(); 
 return body; 
 } 
 
 public static String post(String url, String s) throws IOException, ExecutionException, InterruptedException { 
 AsyncHttpClient http = new AsyncHttpClient(); 
 AsyncHttpClient.BoundRequestBuilder builder = http.preparePost(url); 
 builder.setBodyEncoding(DEFAULT_CHARSET); 
 builder.setBody(s); 
 Future f = builder.execute(); 
 String body = f.get().getResponseBody(DEFAULT_CHARSET); 
 http.close(); 
 return body; 
 } 
 
}

支付工具类pay.java

/** 
 * 支付相关方法 
 * @author andy 
 * 
 */ 
public class Pay { 
 
 // 发货通知接口 
 private static final String DELIVERNOTIFY_URL = "https://api.weixin.qq.com/pay/delivernotify?access_token="; 
 
 /** 
 * 参与 paySign 签名的字段包括:appid、timestamp、noncestr、package 以及 appkey。 
 * 这里 signType 并不参与签名微信的Package参数 
 * @param params 
 * @return 
 * @throws UnsupportedEncodingException 
 */ 
 public static String getPackage(Map params) throws UnsupportedEncodingException { 
 String partnerKey = ConfKit.get("partnerKey"); 
 String partnerId = ConfKit.get("partnerId"); 
 String notifyUrl = ConfKit.get("notify_url"); 
 // 公共参数 
 params.put("bank_type", "WX"); 
 params.put("attach", "yongle"); 
 params.put("partner", partnerId); 
 params.put("notify_url", notifyUrl); 
 params.put("input_charset", "UTF-8"); 
 return packageSign(params, partnerKey); 
 } 
 
 /** 
 * 构造签名 
 * @param params 
 * @param encode 
 * @return 
 * @throws UnsupportedEncodingException 
 */ 
 public static String createSign(Map params, boolean encode) throws UnsupportedEncodingException { 
 Set keysSet = params.keySet(); 
 Object[] keys = keysSet.toArray(); 
 Arrays.sort(keys); 
 StringBuffer temp = new StringBuffer(); 
 boolean first = true; 
 for (Object key : keys) { 
 if (first) { 
 first = false; 
 } else { 
 temp.append("&"); 
 } 
 temp.append(key).append("="); 
 Object value = params.get(key); 
 String valueString = ""; 
 if (null != value) { 
 valueString = value.toString(); 
 } 
 if (encode) { 
 temp.append(URLEncoder.encode(valueString, "UTF-8")); 
 } else { 
 temp.append(valueString); 
 } 
 } 
 return temp.toString(); 
 } 
 
 /** 
 * @param params 
 * @param paternerKey 
 * @return 
 * @throws UnsupportedEncodingException 
 */ 
 private static String packageSign(Map params, String paternerKey) throws UnsupportedEncodingException { 
 String string1 = createSign(params, false); 
 String stringSignTemp = string1 + "&key=" + paternerKey; 
 String signValue = DigestUtils.md5Hex(stringSignTemp).toUpperCase(); 
 String string2 = createSign(params, true); 
 return string2 + "&sign=" + signValue; 
 } 
 
 /** 
 * 支付签名 
 * @param timestamp 
 * @param noncestr 
 * @param packages 
 * @return 
 * @throws UnsupportedEncodingException 
 */ 
 public static String paySign(String timestamp, String noncestr,String packages) throws UnsupportedEncodingException { 
 Map paras = new HashMap(); 
 paras.put("appid", ConfKit.get("AppId")); 
 paras.put("timestamp", timestamp); 
 paras.put("noncestr", noncestr); 
 paras.put("package", packages); 
 paras.put("appkey", ConfKit.get("paySignKey")); 
 // appid、timestamp、noncestr、package 以及 appkey。 
 String string1 = createSign(paras, false); 
 String paySign = DigestUtils.shaHex(string1); 
 return paySign; 
 } 
 
 /** 
 * 支付回调校验签名 
 * @param timestamp 
 * @param noncestr 
 * @param openid 
 * @param issubscribe 
 * @param appsignature 
 * @return 
 * @throws UnsupportedEncodingException 
 */ 
 public static boolean verifySign(long timestamp, 
 String noncestr, String openid, int issubscribe, String appsignature) throws UnsupportedEncodingException { 
 Map paras = new HashMap(); 
 paras.put("appid", ConfKit.get("AppId")); 
 paras.put("appkey", ConfKit.get("paySignKey")); 
 paras.put("timestamp", String.valueOf(timestamp)); 
 paras.put("noncestr", noncestr); 
 paras.put("openid", openid); 
 paras.put("issubscribe", String.valueOf(issubscribe)); 
 // appid、appkey、productid、timestamp、noncestr、openid、issubscribe 
 String string1 = createSign(paras, false); 
 String paySign = DigestUtils.shaHex(string1); 
 return paySign.equalsIgnoreCase(appsignature); 
 } 
 
 /** 
 * 发货通知签名 
 * @param paras 
 * @return 
 * @throws UnsupportedEncodingException 
 * 
 * @参数 appid、appkey、openid、transid、out_trade_no、deliver_timestamp、deliver_status、deliver_msg; 
 */ 
 private static String deliverSign(Map paras) throws UnsupportedEncodingException { 
 paras.put("appkey", ConfKit.get("paySignKey")); 
 String string1 = createSign(paras, false); 
 String paySign = DigestUtils.shaHex(string1); 
 return paySign; 
 } 
 
 
 /** 
 * 发货通知 
 * @param access_token 
 * @param openid 
 * @param transid 
 * @param out_trade_no 
 * @return 
 * @throws IOException 
 * @throws NoSuchProviderException 
 * @throws NoSuchAlgorithmException 
 * @throws KeyManagementException 
 * @throws InterruptedException 
 * @throws ExecutionException 
 */ 
 
 public static boolean delivernotify(String access_token, String openid, String transid, String out_trade_no) throws IOException, ExecutionException, InterruptedException { 
 Map paras = new HashMap(); 
 paras.put("appid", ConfKit.get("AppId")); 
 paras.put("openid", openid); 
 paras.put("transid", transid); 
 paras.put("out_trade_no", out_trade_no); 
 paras.put("deliver_timestamp", (System.currentTimeMillis() / 1000) + ""); 
 paras.put("deliver_status", "1"); 
 paras.put("deliver_msg", "ok"); 
 // 签名 
 String app_signature = deliverSign(paras); 
 paras.put("app_signature", app_signature); 
 paras.put("sign_method", "sha1"); 
 String json = HttpKit.post(DELIVERNOTIFY_URL.concat(access_token), JSONObject.toJSONString(paras)); 
 if (StringUtils.isNotBlank(json)) { 
 JSONObject object = JSONObject.parseObject(json); 
 if (object.containsKey("errcode")) { 
 int errcode = object.getIntValue("errcode"); 
 return errcode == 0; 
 } 
 } 
 return false; 
 } 
}

流转化Tools.java工具类

public final class Tools { 
 
 public static final String inputStream2String(InputStream in) throws UnsupportedEncodingException, IOException{ 
 if(in == null) 
 return ""; 
 
 StringBuffer out = new StringBuffer(); 
 byte[] b = new byte[4096]; 
 for (int n; (n = in.read(b)) != -1;) { 
 out.append(new String(b, 0, n, "UTF-8")); 
 } 
 return out.toString(); 
 } 
 
 public static final boolean checkSignature(String token,String signature,String timestamp,String nonce){ 
 List params = new ArrayList(); 
 params.add(token); 
 params.add(timestamp); 
 params.add(nonce); 
 Collections.sort(params,new Comparator() { 
 @Override 
 public int compare(String o1, String o2) { 
 return o1.compareTo(o2); 
 } 
 }); 
 String temp = params.get(0)+params.get(1)+params.get(2); 
 return SHA1.encode(temp).equals(signature); 
 } 
}

相应前端数据工具WebUtil.java工具类

public class WebUtil { 
 
 public static Object getSessionAttribute(HttpServletRequest req, String key) { 
 Object ret = null; 
 
 try { 
 ret = req.getSession(false).getAttribute(key); 
 } catch (Exception e) { 
 } 
 return ret; 
 } 
 
 public static void response(HttpServletResponse response, String result) { 
 try { 
 response.setContentType("application/json;charset=utf-8"); 
 response.getWriter().write(result); 
 } catch (IOException e) { 
 e.printStackTrace(); 
 } 
 
 } 
 
 public static void response(HttpServletResponse response, ResponseMessage result) { 
 try { 
 response.setContentType("application/json;charset=utf-8"); 
 response.getWriter().write(JsonUtil.objectToJsonNode(result).toString()); 
 } catch (Exception e) { 
 e.printStackTrace(); 
 } 
 } 
 
 public static String packJsonp(String callback, String json) { 
 if (json == null) { 
 json = ""; 
 } 
 if (callback == null || callback.isEmpty()) { 
 return json; 
 } 
 
 return callback + "&&" + callback + &#39;(&#39; + json + &#39;)&#39;; 
 } 
 
 public static String packJsonp(String callback, ResponseMessage response) { 
 String json = null; 
 if (respOnse== null) { 
 json = ""; 
 } else { 
 json = JsonUtil.objectToJsonNode(response).toString(); 
 } 
 if (callback == null || callback.isEmpty()) { 
 return json; 
 } 
 
 return callback + "&&" + callback + &#39;(&#39; + json + &#39;)&#39;; 
 } 
}

Json转换工具JsonUtil.java

public class JsonUtil { 
 
 public static ObjectNode warpJsonNodeResponse(JsonNode obj){ 
 ObjectNode objectNode=createObjectNode(); 
 objectNode.put("code", 1); 
 objectNode.put("response", obj); 
 return objectNode; 
 } 
 
 public static JsonNode objectToJsonNode(Object obj){ 
 try { 
 ObjectMapper objectMapper = new ObjectMapper(); 
 String objJson=objectMapper.writeValueAsString(obj); 
 JsonNode jsOnNode= objectMapper.readTree(objJson); 
 return jsonNode; 
 } catch (JsonProcessingException e) { 
 e.printStackTrace(); 
 } catch (IOException e) { 
 // TODO Auto-generated catch block 
 e.printStackTrace(); 
 } 
 return null; 
 } 
 
}

四、微信H5调起支付

这个url需要后台实现,其实现功能如下:
1、接受微信服务器端发送的支付结果。
2、向微信服务器发送支付结果

具体 参考微信aip(http://pay.weixin.qq.com/wiki/doc/api/index.php?chapter=9_7)
具体代码如下:
4.1、授权向后台发起生成统一下订单页面
wxrepay.jsp

<%@ page language="java" cOntentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> 
<%@ taglib prefix="spring" uri="http://www.springframework.org/tags" %> 
<% 
 String path = request.getContextPath(); 
 String basePath = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + path + "/"; 
 long t = System.currentTimeMillis(); 
%> 
 
 
 
 
 
 
 
 
 
" rel="stylesheet" type="text/css"> 
 
 
 
 

商品

  • beacon 1分钱 1只
  • 测试支付信息

确定购买

首先是请求服务端wxprepay.shml接口,后台向微信支付平台获取支付订单信息,返回前台,wxpay.jsp页面
4.2、确认支付页面

<%@ page language="java" cOntentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> 
<%@ taglib prefix="spring" uri="http://www.springframework.org/tags" %> 
<% 
 String path = request.getContextPath(); 
 String basePath = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + path + "/"; 
 long t = System.currentTimeMillis(); 
%> 
 
 
 
 
 
 
 
 
 
" rel="stylesheet" type="text/css"> 
 
 
 
 

微信js支付测试

  • 测试支付信息

立即支付

4.2、前台涉及到的工具类
Javascript工具类common.js,样式css.css就不贴了没意义。

var path="wxpay"; 
var mainpath = "/wxpay"; 
var appid = "wx16691fcb0523c1a4"; 
var urlpre = "http://www.xxx.com/wxpay/page"; 
var urlhost = "http://www.xxx.com/"; 
 
$(document).ready(function(){ 
 $(".refresher").click(function(){ 
 refresh(); 
 }); 
 $("#goback").click(function(){ 
 goback(); 
 }); 
}); 
function popupMsg(msg){ 
 alert(msg); 
} 
function printUtilViaGet(panel, requestdata, ajaxurl, printfunction){ 
 $.ajax({ 
 type: &#39;GET&#39;, 
 url: ajaxurl, 
 data: requestdata, 
 cache:false, 
 dataType:"json", 
 async: false, 
 success: function(response) { 
 if (response.code){ 
 if (panel != null && panel.length > 0){ 
  $(panel).html(""); 
  if (printfunction != null) 
  $(panel).html(printfunction(response.response)); 
 } 
 return true; 
 } else { 
 //alert(response.reason); 
 } 
 }, 
 error: function(x, e) { 
 //alert("error", x); 
 }, 
 complete: function(x) { 
 //alert("call complete"); 
 } 
 }); 
 return false; 
} 
 
function ajaxUtilViaGet(requestdata, ajaxurl, succFunction, failFunction){ 
 $.ajax({ 
 url: ajaxurl, 
 type: "GET", 
 dataType: "json", 
 cache:false, 
 data: requestdata, 
 async: false, 
 success: function(response) { 
 if (response.code){ 
 if (succFunction != null) 
  succFunction(response.response); 
 } else { 
 if (failFunction != null) 
  failFunction(response.response); 
 } 
 }, 
 error: function(x, e) { 
 //alert("error", x); 
 }, 
 complete: function(x) { 
 } 
 }); 
 return false; 
} 
function printUtil(panel, requestdata, ajaxurl, printfunction, ajaxasync) { 
 if (isEmpty(ajaxasync)) { 
 ajaxasync = false; 
 } 
 $.ajax({ 
 type : &#39;POST&#39;, 
 url : ajaxurl, 
 data : requestdata, 
 cache : false, 
 dataType : "json", 
 async : ajaxasync, 
 success : function(response) { 
 if (response.code) { 
 if (panel != null && panel.length > 0) { 
  $(panel).html(""); 
  if (printfunction != null) 
  $(panel).html(printfunction(response.response)); 
 } 
 return true; 
 } else { 
 // alert(response.reason); 
 } 
 }, 
 error : function(x, e) { 
 // alert("error", x); 
 }, 
 complete : function(x) { 
 // alert("call complete"); 
 } 
 }); 
 return false; 
} 
function appendUtil(panel, requestdata, ajaxurl, printfunction, ajaxasync) { 
 if (isEmpty(ajaxasync)) { 
 ajaxasync = false; 
 } 
 $.ajax({ 
 type : &#39;POST&#39;, 
 url : ajaxurl, 
 data : requestdata, 
 cache : false, 
 dataType : "json", 
 async : ajaxasync, 
 success : function(response) { 
 if (response.code) { 
 if (panel != null && panel.length > 0) { 
  if (printfunction != null) 
  $(panel).append(printfunction(response.response)); 
 } 
 return true; 
 } else { 
 // alert(response.reason); 
 } 
 }, 
 error : function(x, e) { 
 // alert("error", x); 
 }, 
 complete : function(x) { 
 // alert("call complete"); 
 } 
 }); 
 return false; 
} 
 
function ajaxUtilAsync(requestdata, ajaxurl, succFunction, failFunction) { 
 $.ajax({ 
 url : ajaxurl, 
 type : "POST", 
 dataType : "json", 
 cache : false, 
 data : requestdata, 
 async : true, 
 success : function(response) { 
 if (typeof response.code == "number") { 
 if (response.code > 0) { 
  if (succFunction != null) 
  succFunction(response.response); 
 } else { 
  if (failFunction != null) 
  failFunction(response.response); 
 } 
 } else { 
 if (response.result) { 
  if (succFunction != null) 
  succFunction(response.response); 
 } else { 
  if (failFunction != null) 
  failFunction(response.response); 
 } 
 } 
 }, 
 error : function(x, e) { 
 // alert("error", x); 
 }, 
 complete : function(x) { 
 } 
 }); 
 return false; 
} 
 
function ajaxUtil(requestdata, ajaxurl, succFunction, failFunction){ 
 $.ajax({ 
 url: ajaxurl, 
 type: "POST", 
 dataType: "json", 
 cache:false, 
 data: requestdata, 
 async: false, 
 success: function(response) { 
 if (typeof response.code == "number"){ 
 if (response.code > 0){ 
  if (succFunction != null) 
  succFunction(response.response); 
 } else { 
  if (failFunction != null) 
  failFunction(response.response); 
 } 
 } else { 
 if (response.result){ 
  if (succFunction != null) 
  succFunction(response.response); 
 } else { 
  if (failFunction != null) 
  failFunction(response.response); 
 } 
 } 
 }, 
 error: function(x, e) { 
 //alert("error", x); 
 }, 
 complete: function(x) { 
 } 
 }); 
 return false; 
} 
function loadSelection(panel, requestdata, ajaxurl, itemName){ 
 ajaxUtil(requestdata, ajaxurl, function(response){ 
 var list = response.list; 
 for (var i = 0;i"+list[i][itemName]+""); 
 } 
 }, null); 
} 
function ajaxSubmitRefresh(formId) { 
 var hideForm = $(formId); 
 var optiOns= { 
 dataType : "json", 
 beforeSubmit : function() { 
 }, 
 success : function(result) { 
 if (result.result){ 
 showMsg("提交成功"); 
 } else { 
 alert("提交失败!"); 
 } 
 }, 
 error : function(result) { 
 alert("提交失败!"); 
 } 
 }; 
 hideForm.ajaxSubmit(options); 
} 
function ajaxSubmitWithJump(formId, nextPage) { 
 var hideForm = $(formId); 
 var optiOns= { 
 dataType : "json", 
 beforeSubmit : function() { 
 }, 
 success : function(result) { 
 if (result.result){ 
 alert("提交成功"); 
 window.location.href = nextPage; 
 } else { 
 alert("提交失败!"); 
 } 
 }, 
 error : function(result) { 
 alert("提交失败!"); 
 } 
 }; 
 hideForm.ajaxSubmit(options); 
} 
function refresh(){ 
 window.location.href = window.location.href; 
} 
function goback(){ 
 history.go(-1); 
} 
function urlparameter(paras){ 
 var url = location.href; 
 var paraString = url.substring(url.indexOf("?")+1,url.length).split("&"); 
 var paraObj = {}; 
 for (var i=0; j=paraString[i]; i++){ 
 paraObj[j.substring(0,j.indexOf("=")).toLowerCase()] = j.substring(j.indexOf("=")+1,j.length); 
 } 
 var returnValue = paraObj[paras.toLowerCase()]; 
 if(typeof(returnValue)=="undefined"){ 
 return ""; 
 }else{ 
 return returnValue; 
 } 
} 
String.prototype.endWith=function(str){ 
 if(str==null||str==""||this.length==0||str.length>this.length) 
 return false; 
 if(this.substring(this.length-str.length)==str) 
 return true; 
 else 
 return false; 
 return true; 
 }; 
 
 String.prototype.startWith=function(str){ 
 if(str==null||str==""||this.length==0||str.length>this.length) 
 return false; 
 if(this.substr(0,str.length)==str) 
 return true; 
 else 
 return false; 
 return true; 
 }; 
 
 
function getFileUrl(sourceId) { 
 var url = ""; 
 if (navigator.userAgent.indexOf("MSIE")>=1) { // IE 
 url = document.getElementById(sourceId).value; 
 } else if(navigator.userAgent.indexOf("Firefox")>0) { // Firefox 
 url = window.URL.createObjectURL(document.getElementById(sourceId).files.item(0)); 
 } else if(navigator.userAgent.indexOf("Chrome")>0) { // Chrome 
 url = window.URL.createObjectURL(document.getElementById(sourceId).files.item(0)); 
 } 
 return url; 
} 
 
function preImg(sourceId, targetId) { 
 var url = getFileUrl(sourceId); 
 var imgPre = document.getElementById(targetId); 
 imgPre.src = url; 
} 
 
function initWX(){ 
 $.ajax({ 
 url:mainpath+&#39;/wechatjs.do&#39;, 
 type:&#39;POST&#39;, 
 dataType:&#39;json&#39;, 
 async: false, 
 data: {url:location.href.split(&#39;#&#39;)[0]}, 
 success:function(result){ 
 console.log(result); 
 var data=result[&#39;response&#39;][&#39;map&#39;]; 
 if(result[&#39;code&#39;]==1){ 
 wx.config({ 
  debug: false, 
  appId:data[&#39;appId&#39;], 
  timestamp:data[&#39;timestamp&#39;], 
  nonceStr:data[&#39;nonceStr&#39;], 
  signature:data[&#39;signature&#39;], 
  jsApiList: [&#39;onMenuShareTimeline&#39;,&#39;onMenuShareAppMessage&#39;,&#39;getLocation&#39;, &#39;onMenuShareQQ&#39;, &#39;onMenuShareWeibo&#39;] 
 }); 
 }else{ 
 alert("fail to get code"); 
 window.alert(&#39;fail&#39;); 
 }; 
 } 
 }); 
} 
var EARTH_RADIUS = 6378137.0; //单位M 
var PI = Math.PI; 
 
function getRad(d){ 
 return d*PI/180.0; 
} 
function getGreatCircleDistance(lat1,lng1,lat2,lng2){ 
 var radLat1 = getRad(lat1); 
 var radLat2 = getRad(lat2); 
 
 var a = radLat1 - radLat2; 
 var b = getRad(lng1) - getRad(lng2); 
 
 var s = 2*Math.asin(Math.sqrt(Math.pow(Math.sin(a/2),2) + Math.cos(radLat1)*Math.cos(radLat2)*Math.pow(Math.sin(b/2),2))); 
 s = s*EARTH_RADIUS; 
 s = Math.round(s*10000)/10000.0; 
 s = Math.round(s); 
 return s; 
} 
//对Date的扩展,将 Date 转化为指定格式的String 
//月(M)、日(d)、小时(h)、分(m)、秒(s)、季度(q) 可以用 1-2 个占位符, 
//年(y)可以用 1-4 个占位符,毫秒(S)只能用 1 个占位符(是 1-3 位的数字) 
//例子: 
//(new Date()).Format("yyyy-MM-dd hh:mm:ss.S") ==> 2006-07-02 08:09:04.423 
//(new Date()).Format("yyyy-M-d h:m:s.S") ==> 2006-7-2 8:9:4.18 
Date.prototype.format = function(fmt) 
{ //author: meizz 
var o = { 
 "M+" : this.getMonth()+1,  //月份 
 "d+" : this.getDate(),  //日 
 "h+" : this.getHours(),  //小时 
 "m+" : this.getMinutes(),  //分 
 "s+" : this.getSeconds(),  //秒 
 "q+" : Math.floor((this.getMonth()+3)/3), //季度 
 "S" : this.getMilliseconds() //毫秒 
}; 
if(/(y+)/.test(fmt)) 
 fmt=fmt.replace(RegExp.$1, (this.getFullYear()+"").substr(4 - RegExp.$1.length)); 
for(var k in o) 
 if(new RegExp("("+ k +")").test(fmt)) 
fmt = fmt.replace(RegExp.$1, (RegExp.$1.length==1) ? (o[k]) : (("00"+ o[k]).substr((""+ o[k]).length))); 
return fmt; 
}; 
 
//判断为空 
function isEmpty(src){ 
 if(("undefined" == typeof src) || (src == null) || ($.trim(src) == "") ){ 
 return true; 
 } 
 return false; 
} 
 
//判断不为空 
function notEmpty(src){ 
 return !isEmpty(src); 
} 
 
//微信页面授权 snsapi_base方式 
function wecharauto2burl(url) { 
 return "https://open.weixin.qq.com/connect/oauth2/authorize?appid=" + appid 
 + "&redirect_uri=" + encodeURIComponent(url) 
 + "&response_type=code&scope=snsapi_base&state=xybank#wechat_redirect"; 
} 
 
//页面授权针对snsapi_base方式授权的url 
function wecharauto2baseurl(url) { 
 return wecharauto2burl(urlpre+url); 
} 
 
//页面授权针对snsapi_userinfo方式授权的url 
function wecharauto2userinfourl(url) { 
 return "https://open.weixin.qq.com/connect/oauth2/authorize?appid=" + appid 
 + "&redirect_uri=" + encodeURIComponent(urlpre+url) 
 + "&response_type=code&scope=snsapi_userinfo&state=xybank#wechat_redirect"; 
} 
 
 
//微信分享 此方法需放在wx.ready中 
function shareWeChat(title, link, imgUrl, desc){ 
 wx.onMenuShareTimeline({ 
 title: title, // 分享标题 
 link: link, // 分享链接 
 imgUrl: imgUrl, // 分享图标 
 success: function () { 
 // 用户确认分享后执行的回调函数 
 }, 
 cancel: function () { 
 // 用户取消分享后执行的回调函数 
 } 
 }); 
 
 //分享给朋友 
 wx.onMenuShareAppMessage({ 
 title: title, // 分享标题 
 desc: desc, // 分享描述 
 link: link, // 分享链接 
 imgUrl: imgUrl, // 分享图标 
 type: &#39;link&#39;, // 分享类型,music、video或link,不填默认为link 
 dataUrl: &#39;&#39;, // 如果type是music或video,则要提供数据链接,默认为空 
 success: function () { 
 // 用户确认分享后执行的回调函数 
 }, 
 cancel: function () { 
 // 用户取消分享后执行的回调函数 
 } 
 }); 
 
 //分享到QQ 
 wx.onMenuShareQQ({ 
 title: title, // 分享标题 
 desc: desc, // 分享描述 
 link: link, // 分享链接 
 imgUrl: imgUrl, // 分享图标 
 success: function () { 
 // 用户确认分享后执行的回调函数 
 }, 
 cancel: function () { 
 // 用户取消分享后执行的回调函数 
 } 
 }); 
 
 //分享到腾讯微博 
 wx.onMenuShareWeibo({ 
 title: title, // 分享标题 
 desc: desc, // 分享描述 
 link: link, // 分享链接 
 imgUrl: imgUrl, // 分享图标 
 success: function () { 
 // 用户确认分享后执行的回调函数 
 }, 
 cancel: function () { 
 // 用户取消分享后执行的回调函数 
 
 } 
 }); 
}

五、支付结果

公众号调起效果如下:

后续会全部更新微信app支付,微信支付退款,微信企业向个人付款,支付宝相关支付。而且会上传全部代码到csdn资源下载处,尽请关注。

以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注PHP中文网!

以上就是用H5调用支付微信公众号支付的解析的详细内容,更多请关注 第一PHP社区 其它相关文章!


推荐阅读
  • 本文介绍了前端人员必须知道的三个问题,即前端都做哪些事、前端都需要哪些技术,以及前端的发展阶段。初级阶段包括HTML、CSS、JavaScript和jQuery的基础知识。进阶阶段涵盖了面向对象编程、响应式设计、Ajax、HTML5等新兴技术。高级阶段包括架构基础、模块化开发、预编译和前沿规范等内容。此外,还介绍了一些后端服务,如Node.js。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 本文介绍了DataTables插件的官方网站以及其基本特点和使用方法,包括分页处理、数据过滤、数据排序、数据类型检测、列宽度自动适应、CSS定制样式、隐藏列等功能。同时还介绍了其易用性、可扩展性和灵活性,以及国际化和动态创建表格的功能。此外,还提供了参数初始化和延迟加载的示例代码。 ... [详细]
  • 如何实现织梦DedeCms全站伪静态
    本文介绍了如何通过修改织梦DedeCms源代码来实现全站伪静态,以提高管理和SEO效果。全站伪静态可以避免重复URL的问题,同时通过使用mod_rewrite伪静态模块和.htaccess正则表达式,可以更好地适应搜索引擎的需求。文章还提到了一些相关的技术和工具,如Ubuntu、qt编程、tomcat端口、爬虫、php request根目录等。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文介绍了通过ABAP开发往外网发邮件的需求,并提供了配置和代码整理的资料。其中包括了配置SAP邮件服务器的步骤和ABAP写发送邮件代码的过程。通过RZ10配置参数和icm/server_port_1的设定,可以实现向Sap User和外部邮件发送邮件的功能。希望对需要的开发人员有帮助。摘要长度:184字。 ... [详细]
  • 在springmvc框架中,前台ajax调用方法,对图片批量下载,如何弹出提示保存位置选框?Controller方法 ... [详细]
  • 一句话解决高并发的核心原则
    本文介绍了解决高并发的核心原则,即将用户访问请求尽量往前推,避免访问CDN、静态服务器、动态服务器、数据库和存储,从而实现高性能、高并发、高可扩展的网站架构。同时提到了Google的成功案例,以及适用于千万级别PV站和亿级PV网站的架构层次。 ... [详细]
  • 在Android中解析Gson解析json数据是很方便快捷的,可以直接将json数据解析成java对象或者集合。使用Gson解析json成对象时,默认将json里对应字段的值解析到java对象里对应字段的属性里面。然而,当我们自己定义的java对象里的属性名与json里的字段名不一样时,我们可以使用@SerializedName注解来将对象里的属性跟json里字段对应值匹配起来。本文介绍了使用@SerializedName注解解析json数据的方法,并给出了具体的使用示例。 ... [详细]
  • 本文介绍了如何使用jQuery和AJAX来实现动态更新两个div的方法。通过调用PHP文件并返回JSON字符串,可以将不同的文本分别插入到两个div中,从而实现页面的动态更新。 ... [详细]
  • express工程中的json调用方法
    本文介绍了在express工程中如何调用json数据,包括建立app.js文件、创建数据接口以及获取全部数据和typeid为1的数据的方法。 ... [详细]
  • 本文介绍了Java后台Jsonp处理方法及其应用场景。首先解释了Jsonp是一个非官方的协议,它允许在服务器端通过Script tags返回至客户端,并通过javascript callback的形式实现跨域访问。然后介绍了JSON系统开发方法,它是一种面向数据结构的分析和设计方法,以活动为中心,将一连串的活动顺序组合成一个完整的工作进程。接着给出了一个客户端示例代码,使用了jQuery的ajax方法请求一个Jsonp数据。 ... [详细]
  • 目录浏览漏洞与目录遍历漏洞的危害及修复方法
    本文讨论了目录浏览漏洞与目录遍历漏洞的危害,包括网站结构暴露、隐秘文件访问等。同时介绍了检测方法,如使用漏洞扫描器和搜索关键词。最后提供了针对常见中间件的修复方式,包括关闭目录浏览功能。对于保护网站安全具有一定的参考价值。 ... [详细]
  • PHP图片截取方法及应用实例
    本文介绍了使用PHP动态切割JPEG图片的方法,并提供了应用实例,包括截取视频图、提取文章内容中的图片地址、裁切图片等问题。详细介绍了相关的PHP函数和参数的使用,以及图片切割的具体步骤。同时,还提供了一些注意事项和优化建议。通过本文的学习,读者可以掌握PHP图片截取的技巧,实现自己的需求。 ... [详细]
  • 从零基础到精通的前台学习路线
    随着互联网的发展,前台开发工程师成为市场上非常抢手的人才。本文介绍了从零基础到精通前台开发的学习路线,包括学习HTML、CSS、JavaScript等基础知识和常用工具的使用。通过循序渐进的学习,可以掌握前台开发的基本技能,并有能力找到一份月薪8000以上的工作。 ... [详细]
author-avatar
爱与嫉妒咋
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有