1. 程式人生 > >微信公眾號支付H5呼叫支付詳解

微信公眾號支付H5呼叫支付詳解

分享一下我老師大神的人工智慧教程吧。零基礎,通俗易懂!風趣幽默!http://www.captainbed.net/

也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!

                       

原文地址:http://blog.csdn.net/fengshizty/article/details/45564685


微信公眾號支付

最近專案需要微信支付,然後看了下微信公眾號支付,,雖然不難,但是細節還是需要注意的,用了大半天時間寫了個demo,並且完整的測試了一下支付流程,下面分享一下微信公眾號支付的經驗。


一、配置公眾號微信支付  

   需要我們配置微信公眾號支付地址和測試白名單。

  

     比如:支付JS頁面的地址為 http://www.xxx.com/shop/pay/

            那此處配置www.xxx.com/shop/pay/


  二、開發流程

     借用微信公眾號支付api(地址 http://pay.weixin.qq.com/wiki/doc/api/index.PHP?chapter=7_4),我們需要開發的為紅色標記出的。如下:

    

 

三、向微信伺服器端下訂單

             呼叫統一下單介面,這樣就能獲取微信支付的prepay_id(http://pay.weixin.qq.com/wiki/doc/api/index.php?chapter=9_1)。

     在呼叫該介面前有幾個欄位是H5支付必須填寫的openid

    3.1 獲取openid

         可以通過網頁授權形式(http://mp.weixin.qq.com/wiki/17/c0f37d5704f0b64713d5d2c37b468d75.html)

       在微信中傳送如下連結

      

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 後臺支付

    程式碼如下,包含預處理訂單,支付訂單等介面。

 

[java]  view plain  copy

 在CODE上檢視程式碼片派生到我的程式碼片


  1. package org.andy.controller;  
  2.   
  3. import java.io.ByteArrayInputStream;  
  4. import java.io.IOException;  
  5. import java.io.InputStream;  
  6. import java.io.UnsupportedEncodingException;  
  7. import java.util.Date;  
  8. import java.util.HashMap;  
  9. import java.util.Iterator;  
  10. import java.util.Map;  
  11. import java.util.Map.Entry;  
  12. import java.util.Random;  
  13.   
  14. import javax.servlet.ServletInputStream;  
  15. import javax.servlet.http.HttpServletRequest;  
  16. import javax.servlet.http.HttpServletResponse;  
  17.   
  18. import org.apache.commons.codec.digest.DigestUtils;  
  19. import org.springframework.stereotype.Controller;  
  20. import org.springframework.web.bind.annotation.RequestMapping;  
  21. import org.xmlpull.v1.XmlPullParser;  
  22. import org.xmlpull.v1.XmlPullParserException;  
  23. import org.xmlpull.v1.XmlPullParserFactory;  
  24.   
  25. import com.fasterxml.jackson.databind.JsonNode;  
  26. import com.gson.oauth.Oauth;  
  27. import com.gson.oauth.Pay;  
  28. import com.gson.util.HttpKit;  
  29. import com.gson.util.Tools;  
  30. import org.andy.util.DatetimeUtil;  
  31. import org.andy.util.JsonUtil;  
  32. import org.andy.util.SessionUtil;  
  33. import org.andy.util.WebUtil;  
  34.   
  35. @Controller  
  36. @RequestMapping(“/pay”)  
  37. public class WXPayController {  
  38.   
  39.     @RequestMapping(value = “wxprepay”)  
  40.     public void jspay(HttpServletRequest request, HttpServletResponse response, String callback) throws Exception {  
  41.         // 獲取openid  
  42.         String openId = SessionUtil.getAtt(request, ”openId”);  
  43.         if (openId == null) {  
  44.             openId = getUserOpenId(request);  
  45.         }  
  46.   
  47.         String appid = ”wx16691fcb0523c1a4”;  
  48.         String partnerid = ”22223670”;  
  49.         String paternerKey = ”fjfjfjfjf1234567FFFFFFFFF1234567”;  
  50.           
  51.         String out_trade_no = getTradeNo();  
  52.         Map<String, String> paraMap = new HashMap<String, String>();  
  53.         paraMap.put(”appid”, appid);  
  54.         paraMap.put(”attach”“測試支付”);  
  55.         paraMap.put(”body”“測試購買Beacon支付”);  
  56.         paraMap.put(”mch_id”, partnerid);  
  57.         paraMap.put(”nonce_str”, create_nonce_str());  
  58.         paraMap.put(”openid”, openId);  
  59.         paraMap.put(”out_trade_no”, out_trade_no);  
  60.         paraMap.put(”spbill_create_ip”, getAddrIp(request));  
  61.         paraMap.put(”total_fee”“1”);  
  62.         paraMap.put(”trade_type”“JSAPI”);  
  63.         paraMap.put(”notify_url”“http://www.xxx.co/wxpay/pay/appPay_notify.shtml”);  
  64.         String sign = getSign(paraMap, paternerKey);  
  65.         paraMap.put(”sign”, sign);  
  66.   
  67.         // 統一下單 https://api.mch.weixin.qq.com/pay/unifiedorder  
  68.         String url = ”https://api.mch.weixin.qq.com/pay/unifiedorder”;  
  69.   
  70.         String xml = ArrayToXml(paraMap, false);  
  71.   
  72.         String xmlStr = HttpKit.post(url, xml);  
  73.   
  74.         // 預付商品id  
  75.         String prepay_id = ”“;  
  76.   
  77.         if (xmlStr.indexOf(“SUCCESS”) != -1) {  
  78.             Map<String, String> map = doXMLParse(xmlStr);  
  79.             prepay_id = (String) map.get(”prepay_id”);  
  80.         }  
  81.   
  82.         Map<String, String> payMap = new HashMap<String, String>();  
  83.         payMap.put(”appId”, appid);  
  84.         payMap.put(”timeStamp”, create_timestamp());  
  85.         payMap.put(”nonceStr”, create_nonce_str());  
  86.         payMap.put(”signType”“MD5”);  
  87.         payMap.put(”package”“prepay_id=” + prepay_id);  
  88.         String paySign = getSign(payMap, paternerKey);  
  89.           
  90.         payMap.put(”pg”, prepay_id);  
  91.         payMap.put(”paySign”, paySign);  
  92.           
  93.           
  94.         WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(payMap)).toString()));  
  95.     }  
  96.       
  97.     @RequestMapping(value = “appPay”)  
  98.     public void appPay(HttpServletRequest request, HttpServletResponse response, String body, String detail, String total_fee, String  spbill_create_ip,   
  99.             String notify_url, String trade_type, String callback) throws Exception {  
  100.   
  101.         String appid = ”wx16691fcb0523c1a4”;  
  102.         String partnerid = ”22223670”;  
  103.         String paternerKey = ”fjfjfjfjf1234567FFFFFFFFF1234567”;  
  104.           
  105.         String out_trade_no = getTradeNo();  
  106.         Map<String, String> paraMap = new HashMap<String, String>();  
  107.         paraMap.put(”appid”, appid);  
  108.         paraMap.put(”body”, body);  
  109.         paraMap.put(”mch_id”, partnerid);  
  110.         paraMap.put(”nonce_str”, create_nonce_str());  
  111.         paraMap.put(”out_trade_no”, out_trade_no);  
  112.         paraMap.put(”spbill_create_ip”, spbill_create_ip);  
  113.         paraMap.put(”total_fee”, total_fee);  
  114.         paraMap.put(”trade_type”, trade_type);  
  115.         paraMap.put(”notify_url”, notify_url);  
  116.         String sign = getSign(paraMap, paternerKey);  
  117.         paraMap.put(”sign”, sign);  
  118.   
  119.         // 統一下單 https://api.mch.weixin.qq.com/pay/unifiedorder  
  120.         String url = ”https://api.mch.weixin.qq.com/pay/unifiedorder”;  
  121.   
  122.         String xml = ArrayToXml(paraMap, false);  
  123.   
  124.         String xmlStr = HttpKit.post(url, xml);  
  125.   
  126.         // 預付商品id  
  127.         String prepay_id = ”“;  
  128.   
  129.         Map<String, String> map = doXMLParse(xmlStr);  
  130.         if (xmlStr.indexOf(“SUCCESS”) != -1) {  
  131.             prepay_id = (String) map.get(”prepay_id”);  
  132.         }  
  133.   
  134.         String result_code = map.get(”result_code”);  
  135.         String err_code_des = map.get(”err_code_des”);  
  136.         Map<String, String> payMap = new HashMap<String, String>();  
  137.         payMap.put(”appid”, appid);  
  138.         payMap.put(”partnerid”, partnerid);  
  139.         payMap.put(”prepayid”, prepay_id);  
  140.         payMap.put(”package”“Sign=WXPay”);  
  141.         payMap.put(”noncestr”, create_nonce_str());  
  142.         payMap.put(”timestamp”, create_timestamp());  
  143.         String paySign = getSign(payMap, paternerKey);  
  144.           
  145.         payMap.put(”sign”, paySign);  
  146.         payMap.put(”result_code”, result_code);  
  147.         payMap.put(”err_code_des”, err_code_des);  
  148.           
  149.           
  150.         WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(payMap)).toString()));  
  151.     }  
  152.       
  153.     @RequestMapping(“/appPay_notify”)  
  154.     public void appPayNotify(HttpServletRequest request, HttpServletResponse response) throws Exception{  
  155.         //String xml = ”<xml><appid><![CDATA[wxb4dc385f953b356e]]></appid><bank_type><![CDATA[CCB_CREDIT]]></bank_type><cash_fee><![CDATA[1]]></cash_fee><fee_type><![CDATA[CNY]]></fee_type><is_subscribe><![CDATA[Y]]></is_subscribe><mch_id><![CDATA[1228442802]]></mch_id><nonce_str><![CDATA[1002477130]]></nonce_str><openid><![CDATA[o-HREuJzRr3moMvv990VdfnQ8x4k]]></openid><out_trade_no><![CDATA[1000000000051249]]></out_trade_no><result_code><![CDATA[SUCCESS]]></result_code><return_code><![CDATA[SUCCESS]]></return_code><sign><![CDATA[1269E03E43F2B8C388A414EDAE185CEE]]></sign><time_end><![CDATA[20150324100405]]></time_end><total_fee>1</total_fee><trade_type><![CDATA[JSAPI]]></trade_type><transaction_id><![CDATA[1009530574201503240036299496]]></transaction_id></xml>”;  
  156.         response.setCharacterEncoding(”UTF-8”);  
  157.         response.setContentType(”text/xml”);  
  158.         ServletInputStream in = request.getInputStream();  
  159.         String xmlMsg = Tools.inputStream2String(in);  
  160.           
  161.         Map<String, String> map = doXMLParse(xmlMsg);  
  162.         String return_code = map.get(”return_code”);  
  163.         String return_msg = map.get(”return_msg”);  
  164.           
  165.         map = new HashMap<String, String>();  
  166.         map.put(”return_code”, return_code);  
  167.         map.put(”return_msg”, return_msg);  
  168.           
  169.         //響應xml  
  170.         String resXml = ArrayToXml(map, true);  
  171.         response.getWriter().write(resXml);  
  172.     }  
  173.   
  174.     @RequestMapping(“/orderquery.do”)  
  175.     public void orderquery(HttpServletRequest request, HttpServletResponse response, String transaction_id, String out_trade_no, String callback) throws Exception{  
  176.           
  177.         String url = ”https://api.mch.weixin.qq.com/pay/orderquery”;  
  178.   
  179.         String appid = ”wx16691fcb0523c1a4”;  
  180.         String partnerid = ”22223670”;  
  181.         String paternerKey = ”fjfjfjfjf1234567FFFFFFFFF1234567”;  
  182.           
  183.         Map<String, String> map = new HashMap<String, String>();  
  184.         map.put(”appid”, appid);  
  185.         map.put(”mch_id”, partnerid);  
  186.         if(transaction_id != null && !transaction_id.equals(“”)){  
  187.             map.put(”transaction_id”, transaction_id);  
  188.         }else {  
  189.             map.put(”out_trade_no”, out_trade_no);  
  190.         }  
  191.         map.put(”nonce_str”, create_nonce_str());  
  192.         String paySign = getSign(map, paternerKey);  
  193.         map.put(”sign”, paySign);  
  194.           
  195.         String xml = ArrayToXml(map, false);  
  196.         String xmlStr = HttpKit.post(url, xml);  
  197.           
  198.         Map<String, String> orderMap = doXMLParse(xmlStr);  
  199.           
  200.           
  201.         WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(orderMap)).toString()));  
  202.     }  
  203.       
  204.     /** 
  205.      * map轉成xml 
  206.      *  
  207.      * @param arr 
  208.      * @return 
  209.      */  
  210.     public String ArrayToXml(Map<String, String> parm, boolean isAddCDATA) {  
  211.         StringBuffer strbuff = new StringBuffer(<xml>);  
  212.         if (parm != null ) {  
  213.             for (Entry<String, String> entry : parm.entrySet()) {  
  214.                 strbuff.append(”<”).append(entry.getKey()).append(“>”);  
  215.                 if (isAddCDATA) {  
  216.                     strbuff.append(<![CDATA[).append(entry.getValue()).append(]]>);  
  217.                 }else {  
  218.                     strbuff.append(entry.getValue());  
  219.                 }  
  220.                 strbuff.append(”<”).append(entry.getKey()).append(“>”);  
  221.             }  
  222.         }  
  223.         return strbuff.append(</xml>).toString();  
  224.     }  
  225.   
  226.     // 獲取openId  
  227.     private String getUserOpenId(HttpServletRequest request) throws Exception {  
  228.         String code = request.getParameter(”code”);  
  229.         if (code == null) {  
  230.             String openId = request.getParameter(”openId”);  
  231.             return openId;  
  232.         }  
  233.         Oauth o = new Oauth();  
  234.         String token = o.getToken(code);  
  235.         JsonNode node = JsonUtil.StringToJsonNode(token);  
  236.         String openId = node.get(”openid”).asText();  
  237.         return openId;  
  238.     }  
  239.   
  240.     private String create_nonce_str() {  
  241.             String chars = ”abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789”;  
  242.             String res = ”“;  
  243.             for (int i = 0; i < 16; i++) {  
  244.                 Random rd = new Random();  
  245.                 res += chars.charAt(rd.nextInt(chars.length() - 1));  
  246.             }  
  247.             return res;  
  248.     }  
  249.       
  250.     private String getAddrIp(HttpServletRequest request){  
  251.         return request.getRemoteAddr();  
  252.     }  
  253.   
  254.     private String create_timestamp() {  
  255.         return Long.toString(System.currentTimeMillis() / 1000);  
  256.     }  
  257.       
  258.     private String getTradeNo(){  
  259.         String timestamp = DatetimeUtil.formatDate(new Date(), DatetimeUtil.DATETIME_PATTERN);  
  260.         return “HZNO” + timestamp;  
  261.     }  
  262.       
  263.     private String getSign(Map<String, String> params, String paternerKey )  
  264.             throws UnsupportedEncodingException {  
  265.         String string1 = Pay.createSign(params, false);  
  266.         String stringSignTemp = string1 + ”&key=” + paternerKey;  
  267.         String signValue = DigestUtils.md5Hex(stringSignTemp).toUpperCase();  
  268.         return  signValue;  
  269.     }  
  270.   
  271.     private Map<String, String> doXMLParse(String xml)  
  272.             throws XmlPullParserException, IOException {  
  273.   
  274.         InputStream inputStream = new ByteArrayInputStream(xml.getBytes());  
  275.   
  276.         Map<String, String> map = null;  
  277.   
  278.         XmlPullParser pullParser = XmlPullParserFactory.newInstance()  
  279.                 .newPullParser();  
  280.   
  281.         pullParser.setInput(inputStream, ”UTF-8”); // 為xml設定要解析的xml資料  
  282.   
  283.         int eventType = pullParser.getEventType();  
  284.   
  285.         while (eventType != XmlPullParser.END_DOCUMENT) {  
  286.             switch (eventType) {  
  287.             case XmlPullParser.START_DOCUMENT:  
  288.                 map = new HashMap<String, String>();  
  289.                 break;  
  290.   
  291.             case XmlPullParser.START_TAG:  
  292.                 String key = pullParser.getName();  
  293.                 if (key.equals(“xml”))  
  294.                     break;  
  295.   
  296.                 String value = pullParser.nextText();  
  297.                 map.put(key, value);  
  298.   
  299.                 break;  
  300.   
  301.             case XmlPullParser.END_TAG:  
  302.                 break;  
  303.   
  304.             }  
  305.   
  306.             eventType = pullParser.next();  
  307.   
  308.         }  
  309.   
  310.         return map;  
  311.     }  
  312.   
  313. }  




   



   

  wxprepay.shtm介面是預處理訂單介面向微信伺服器下訂單。

  appPay.shtml介面是支付介面。

  appPay_notify.shtml介面是微信支付後非同步通知結果介面。

  orderquery.shtml介面是訂單查詢介面

3.3、涉及到的工具類

    SessionUtil.Java工具類

[java]  view plain  copy   在CODE上檢視程式碼片 派生到我的程式碼片
  1. package org.andy.util;  
  2.   
  3. import javax.servlet.http.HttpServletRequest;  
  4.   
  5.   
  6. public class SessionUtil {  
  7.     public static void addAtt(HttpServletRequest request, String key, Object value){  
  8.         request.getSession().setAttribute(key, value);  
  9.     }  
  10.       
  11.     public static void removeAtt(HttpServletRequest request, String key){  
  12.         request.getSession().removeAttribute(key);  
  13.     }  
  14.       
  15.     public static String getAtt(HttpServletRequest request, String key){  
  16.         return (String)request.getSession().getAttribute(key);  
  17.     }  
  18.       
  19.     public static Object getAttObj(HttpServletRequest request, String key){  
  20.         return request.getSession().getAttribute(key);  
  21.     }  
  22.       
  23.     public static String optAtt(HttpServletRequest request, String key, String value){  
  24.         String r = (String)request.getSession().getAttribute(key);  
  25.         if (r == null){  
  26.             r = value;  
  27.         }  
  28.         return r;  
  29.     }  
  30.       
  31. }  

HttpKit 網路請求工具類

[java]  view plain  copy

 在CODE上檢視程式碼片派生到我的程式碼片


  1. /** 
  2.  * https 請求 微信為https的請求 
  3.  * 
  4.  * @author andy 
  5.  * @date 2015-10-9 下午2:40:19 
  6.  */   
  7. public class HttpKit {  
  8.     private static final String DEFAULT_CHARSET = “UTF-8”;  
  9.     /** 
  10.      * @return 返回型別: 
  11.      * @throws IOException 
  12.      * @throws UnsupportedEncodingException 
  13.      * @throws NoSuchProviderException 
  14.      * @throws NoSuchAlgorithmException 
  15.      * @throws KeyManagementException 
  16.      * @description 功能描述: get 請求 
  17.      */  
  18.     public static String get(String url, Map<String, String> params, Map<String, String> headers) throws