1. 程式人生 > >呼叫微信和支付寶第三方介面方法總結

呼叫微信和支付寶第三方介面方法總結

  1. <span style="white-space:pre">  </span>    //判斷證書  
  2.             if (resHandler.getParameter("sign").equals(  
  3.                 WeixinSignUtil.createSign("utf-8",  
  4.                     resHandler.getAllParameters(), securit)))  
  5.             {  
  6.                 // 商戶訂單號  
  7.                 String out_trade_no = resHandler.getParameter("out_trade_no");  
  8.                 String totalFee = resHandler.getParameter("total_fee");  
  9.                 System.out.println("out_trade_no:" + out_trade_no);  
  10.                 Map<String, Object> map = new HashMap<String, Object>();  
  11.                 String result_code = resHandler.getParameter("result_code");  
  12.                 if (result_code.equals("SUCCESS"))  
  13.                 {  
  14.                 <span style="font-family: Arial, Helvetica, sans-serif;">// 金額,以分為單位</span><pre name="code" class="java"><span style="white-space:pre">     </span>String total_fee = resHandler.getParameter("total_fee");</pre><pre name="code" class="java"><span style="white-space:pre">       </span>//進行支付後操作...</pre><span style="white-space:pre"></span>} else { return resHandler.getParameter("err_code_des"); } // 給微信伺服器返回success 否則30分鐘通知8次 return "success"; } else { System.out.println("通知簽名驗證失敗"); resHandler.sendToCFT("fail");  
  15.  response.setCharacterEncoding("utf-8"); return "FAIL"; } } catch (Exception e) { return "FAIL"; } }  
  16. <pre></pre>  
  17. <p></p>  
  18. <pre></pre>  
  19. 這是微信呼叫我們定義的請求來主動返回支付結果,但我們也可以寫介面主動呼叫微信來獲取訂單資訊  
  20. <p></p>  
  21. <p><span style="font-size:14px"><span style="font-size:14px">這裡需要ios或者安卓呼叫我們java後臺寫的介面來獲取支付結果:</span></span></p>  
  22. <p><span style="font-size:14px"><span style="font-size:14px">介面如下:</span></span></p>  
  23. <p><span style="font-size:14px"><span style="font-size:14px"></span></span></p>  
  24. <pre name="code" class="java">/** 
  25.      * 客戶端微信支付成功後查詢後臺支付結果介面 
  26.      *  
  27.      * @param req 
  28.      * @param rsp 
  29.      * @return String 返回json字串 如果有違例,請使用@exception/throws [違例型別] 
  30.      *         [違例說明:異常的註釋必須說明該異常的含義及什麼條件下丟擲該 
  31.      * @throws Exception 
  32.      * @see [類、類#方法、類#成員] 
  33.      */  
  34.     @RequestMapping(value = "/interface/payStatus", method = RequestMethod.POST, produces = "text/html;charset=UTF-8")  
  35.     @ResponseBody  
  36.     public String getPayStatus(HttpServletRequest req, HttpServletResponse rsp)  
  37.         throws Exception  
  38.     {  
  39.         Map<String, Object> resultMap = new HashMap<String, Object>();  
  40.         try  
  41.         {  
  42.             URL postUrl = new URL(  
  43.                 "https://api.mch.weixin.qq.com/pay/orderquery");  
  44.             // 獲取請求內容  
  45.             String requestContent = getRequestContent(req);  
  46.             // 日誌  
  47.             interfaceLogger.info("request payStatus:" + requestContent);  
  48.             // 解析引數  
  49.             requestParam = parseAndCheckParam(requestContent, resultMap,  
  50.                 requestParam, systemParam.getSystemKey());  
  51.             // 校驗引數  
  52.             checkStatusParam(resultMap);  
  53.             // 校驗token及獲取使用者資訊  
  54.             Map<String, Object> userInfo = checkTokenAndGetUserInfo(  
  55.                 tokenService, appuserManager, requestParam, resultMap);  
  56.             String user_id = (String)userInfo.get("userId");  
  57.             String totalFee = (String)requestParam.get("totalFee");  
  58.             String type = (String)requestParam.get("orderType"); // 獲取訂單型別引數  
  59.             if (null == user_id)  
  60.             {  
  61.                 resultMap.put("resultCode",  
  62.                     ErrorCodeConstants.INVALID_TOKEN_ERROR);  
  63.             }  
  64.             SysParamServiceImpl sysParam = (SysParamServiceImpl)SpringContextHolder  
  65.                 .getBean(SysParamServiceImpl.class);  
  66.             HttpsURLConnection con = (HttpsURLConnection)postUrl  
  67.                 .openConnection();// 開啟連線  
  68.             con.setRequestMethod("POST");// post方式提交  
  69.             con.setDoOutput(true);// 開啟讀寫屬性,預設均為false  
  70.             con.setDoInput(true);  
  71.             con.setUseCaches(false);// Post請求不能使用快取  
  72.             con.setInstanceFollowRedirects(true);  
  73.             // DataOutputStream out = new  
  74.             // DataOutputStream(con.getOutputStream());  
  75.             PrintWriter out = new PrintWriter(con.getOutputStream());  
  76.             Map<String, Object> map = new HashMap<String, Object>();  
  77.             // 相關引數,可以定義一個Map做成動態的  
  78.             map.put("appid", sysParam.getStringParamByKey("c.wxpay.appid")); // appID  
  79.             map.put("mch_id", sysParam.getStringParamByKey("c.wxpay.partnerid")); // 商戶號  
  80.             map.put("nonce_str", RandomUtil.CreateRandom(32)); // 隨機數  
  81.             String orderId = requestParam.get("outTradeNo").toString();  
  82.             map.put("out_trade_no", orderId); // 訂單號  
  83.             // String mo="3000";  
  84.             // insertMoneyAccounting(orderId,mo); //利潤分配  
  85.             // 轉成sortedMap,微信支付後臺生成sign需要排序過的map  
  86.             SortedMap<String, Object> sort = new TreeMap<String, Object>(map);  
  87.             String secrit = sysParam.getStringParamByKey("c.wxpay.secret");  
  88.             // 生成sign  
  89.             String sign = WeixinSignUtil.createSign("UTF-8", sort, secrit);  
  90.             // 把sign放入map中  
  91.             map.put("sign", sign); // 簽名  
  92.             Document doc = DocumentHelper.createDocument();  
  93.             Element body = DocumentHelper.createElement("xml");  
  94.             buildMap2xmlBody(body, map);  
  95.             doc.add(body);  
  96.             // 傳送請求  
  97.             // out.writeUTF(doc.asXML());  
  98.             String outStr = doc.asXML();  
  99.             out.print(outStr);  
  100.             System.out.println(doc.asXML());  
  101.             out.flush();  
  102.             out.close();  
  103.             // 接收資料  
  104.             BufferedReader reader = new BufferedReader(new InputStreamReader(  
  105.                 con.getInputStream(), "UTF-8"));  
  106.             String line;  
  107.             StringBuffer responseText = new StringBuffer();  
  108.             while ( (line = reader.readLine()) != null)  
  109.             {  
  110.                 responseText.append(line).append("\r\n");  
  111.             }  
  112.             reader.close();  
  113.             con.disconnect();  
  114.             String resXml = responseText.toString() + ""; // 獲取從微信付款的結果  
  115.                                                           // ,以String型別的xml格式返回  
  116.             System.out.println("result:" + resXml);  
  117.             // 解析xml  
  118.             Map<String, Object> m = new HashMap<String, Object>();  
  119.             Document d = DocumentHelper.parseText(resXml);  
  120.             Element rootElement = d.getRootElement();  
  121.             XmlToBean.ele2map(m, rootElement);  
  122.             String str = "";  
  123.             if (m.containsKey("trade_state"))  
  124.             {  
  125.                 str = m.get("trade_state").toString();  
  126.                 str = str.replace("{trade_state=", "");  
  127.                 str = str.substring(0, str.length() - 1);  
  128.                 if ("SUCCESS" == str || "SUCCESS".equals(str))  
  129.                 {  
  130.                     if ("0".equals(type)) {  
  131.                         //支付成功後操作...  
  132.                     }  
  133.                 }  
  134.                 else if ("REFUND" == str || "REFUND".equals(str))  
  135.                 {  
  136.                     str = "1"; // 轉入退款  
  137.                 }  
  138.                 else if ("NOTPAY" == str || "NOTPAY".equals(str))  
  139.                 {  
  140.                     str = "2"; // 未支付  
  141.                 }  
  142.                 else if ("CLOSED" == str || "CLOSED".equals(str))  
  143.                 {  
  144.                     str = "3"; // 已關閉  
  145.                 }  
  146.                 else if ("REVOKED" == str || "REVOKED".equals(str))  
  147.                 {  
  148.                     str = "4"; // 已撤銷  
  149.                 }  
  150.                 else if ("USERPAYING" == str || "USERPAYING".equals(str))  
  151.                 {  
  152.                     str = "5"; // 使用者支付中  
  153.                 }  
  154.                 else if ("PAYERROR" == str || "PAYERROR".equals(str))  
  155.                 {  
  156.                     str = "6"; // 支付失敗  
  157.                 }  
  158.                 resultMap.put("status", str);  
  159.                 resultMap.put("resultCode",  
  160.                     ErrorCodeConstants.RESULT_CODE_SUCCESS);  
  161.             }  
  162.         }  
  163.         catch (Exception e)  
  164.         {  
  165.             logger.error("loginOut:system exception!", e);  
  166.             resultMap.clear();  
  167.             resultMap.put("resultCode", ErrorCodeConstants.SYSTEM_ERROR);  
  168.         }  
  169.         interfaceLogger.info("response payStatus:" + resultMap.toString());  
  170.         return JsonUtil.Object2EncodeJsonSting(resultMap,  
  171.             systemParam.getSystemKey());  
  172.     }</pre><br>  
  173. requestParam.get("XXX"),XXX是客戶端呼叫我們介面給我們傳的引數,需要給我們totalFee(金額),orderType(訂單型別)等<br>  
  174. <p></p>  
  175. <p><span style="font-size:14px"><span style="font-size:14px">然後我們返回支付結果給客戶端</span></span></p>  
  176. <p><span style="font-size:14px; color:#ff0000">二、支付寶支付:</span></p>  
  177. <p><span style="font-size:14px">支付寶支付流程上和微信支付稍微有所差別,支付寶不需要呼叫下單結果,但是也需要一個類似於微信prepayId的欄位來入庫,作用上跟prepayId差不多,名稱可以自己定義,我這邊暫時為aliPayInfo:</span></p>  
  178. <p><span style="font-size:14px">首先看js程式碼:</span></p>  
  179. <p><span style="font-size:14px"></span></p>  
  180. <pre name="code" class="javascript">//支付寶支付   
  181.             function aliPay(outTradeNo){  
  182.                 var orderType="0";  
  183.                 var info = $("#aliPayInfo").val();//訂單資訊   
  184.                 var totalAmount = $("#moeny").val();//總金額  
  185.                 var orderName = $("#orderName").val();//訂單名稱  
  186.                 //type 0:線路 1:景點  2:酒店 3:商城 4:VIP  
  187.                 var prams = {  
  188.                         aliPayInfo : info,  
  189.                         type : orderType,  
  190.                         orderNumber : outTradeNo,  
  191.                         money : totalAmount,  
  192.                         orderName : orderName  
  193.                 };  
  194.                 //先調支付寶確定是否下單 否則返回支付寶會話資訊  
  195.                 $.ajax({  
  196.                     url : '<%=basePath%>client/hotel/foundZFBOrder.do',  
  197.                     type : "post",  
  198.                     data : prams,  
  199.                     cache : false,  
  200.                     dataType : "json",  
  201.                     success : function(data)  
  202.                     {  
  203.                         //alert("info +===="+data.info);  
  204.                         if(data.isSuccess){  
  205.                             if(isAndroid){  
  206.                                 ClientInterface.aliPay(data.info,outTradeNo,orderType,totalAmount);  
  207.                             }else if(isiOS){  
  208.                                 window.location.href='http://localhost/aliPay/'+data.info+';'+outTradeNo+";"+orderType+";"+totalAmount;  
  209.                             }  
  210.                         }else{  
  211.                             var title = "支付寶下單失敗";  
  212.                             prompt(title);  
  213.                         }  
  214.                     },   
  215.                     error : function(data){  
  216.                         var title = "支付寶下單失敗";  
  217.                         prompt(title);  
  218.                     }  
  219.                 });  
  220.             }</pre><br>  
  221. <pre name="code" class="javascript"></pre>頁面上的aliPayInfo可以為空,因為第一次去呼叫支付寶時這個為空,如果不支付,但是還是入庫了,需要去資料庫裡查一下,那麼共用這個頁面的話aliPayInfo就不為空<p></p><p><span style="font-size:14px">然後去後臺寫這個ajax請求</span></p><p><span style="font-size:14px"></span></p><pre name="code" class="java">@RequestMapping(value="/foundZFBOrder", produces = "application/json;charset=UTF-8")  
  222.     @ResponseBody  
  223.     public Map<String, Object> foundZFBOrder(HttpServletRequest request){  
  224.         //支付寶資訊  
  225.         String aliPayInfo = request.getParameter("aliPayInfo");  
  226.         //金額  
  227.         String money = request.getParameter("money");  
  228.         //訂單名稱  
  229.         String orderName = request.getParameter("orderName");  
  230.         //訂單編號  
  231.         String orderNumber = request.getParameter("orderNumber");  
  232.         //訂單型別(0:線路 1:景點  2:酒店 3:商城 4:VIP)  
  233.         String type = request.getParameter("type");  
  234.         boolean isSuccess = false;  
  235.         //判斷是否生成過支付寶資訊  
  236.         if(StringUtils.isBlank(aliPayInfo)){  
  237.             //生成支付寶資訊  
  238.             aliPayInfo = AliPayUtil.orderInfo(money, orderName, orderNumber, type);  
  239.             //修改表中支付寶資訊  
  240.             if(StringUtils.isNotBlank(aliPayInfo)){  
  241.                 isSuccess = true;  
  242.                 Map<String, Object> map = new HashMap<String, Object>();  
  243.                 map.put("aliPayInfo", aliPayInfo);  
  244.                 map.put("type", type);  
  245.                 map.put("orderNumber", orderNumber);  
  246.                 try {  
  247.                     travelOrderService.updateOrderInfoByType(map);  
  248.                 }  
  249.                 catch (Exception e) {  
  250.                     e.printStackTrace();  
  251.                 }  
  252.             }  
  253.             /** 列印支付寶響應日誌  */  
  254.             if (interfaceLogger.isInfoEnabled()) {  
  255.                 interfaceLogger.info("[HotelController][foundZFBOrder]ZFB " + table + " result :["  
  256.                                      + JSON.toJSONString(aliPayInfo) + "]");  
  257.             }  
  258.         }  
  259.         Map<String, Object> map = new HashMap<String, Object>();  
  260.         map.put("info", aliPayInfo);  
  261.         map.put("isSuccess", isSuccess);  
  262.         return map;  
  263.     }</pre><br>  
  264. <pre name="code" class="java"></pre>這裡的 aliPayInfo = AliPayUtil.orderInfo(money, orderName, orderNumber, type);<p></p><p><span style="font-size:14px">類似於微信的下單介面(<span style="font-size:14px">aliPayInfo </span>需要入庫),但不是下單介面,因為支付寶支付沒有下單介面,詳情可以檢視支付寶支付文件</span></p><p><span style="font-size:14px">下面繼續寫orderInfo方法:</span></p><pre name="code" class="java">/** 
  265.      *  
  266.      * @param amount 訂單金額 
  267.      * @param subject 訂單標題 
  268.      * @param body 訂單描述 
  269.      * @param outTradeNo訂單號 
  270.      * @return 
  271.      */  
  272.     public static String orderInfo(String amount,String subject,String outTradeNo,String type) {  
  273.         //獲取appid  
  274.         String appId=Const.APPID;  
  275.         Map<String, String> authInfoMap = OrderInfoUtil.buildOrderParamMap(appId,amount,subject,outTradeNo,type);  
  276.         String info = OrderInfoUtil.buildOrderParam(authInfoMap);  
  277.         String rsaPrivate =Const.RSAPRIVATE; //獲取商戶金鑰  
  278.         String sign = OrderInfoUtil.getSign(authInfoMap, rsaPrivate);  
  279.         final String orderInfo = info + "&" + sign;  
  280.         return orderInfo;  
  281.     }</pre><br>  
  282. <span style="font-size:18px">這裡和微信一樣需要支付寶商戶的appid和RSAPRIVATE提供給我們開發者,</span><br>  
  283. <p style="font-size:14px">然後編寫支付寶支付工具類:</p>  
  284. <p style="font-size:14px"></p>  
  285. <pre name="code" class="java">public class OrderInfoUtil {  
  286.     /** 
  287.      * 構造授權引數列表 
  288.      *  
  289.      * @param pid 
  290.      * @param app_id 
  291.      * @param target_id 
  292.      * @return 
  293.      */  
  294.     public static Map<String, String> buildAuthInfoMap(String pid, String app_id, String target_id) {  
  295.         Map<String, String> keyValues = new HashMap<String, String>();  
  296.         // 商戶簽約拿到的app_id,如:2013081700024223  
  297.         keyValues.put("app_id", app_id);  
  298.         // 商戶簽約拿到的pid,如:2088102123816631  
  299.         keyValues.put("pid", pid);  
  300.         // 服務介面名稱, 固定值  
  301.         keyValues.put("apiname", "com.alipay.account.auth");  
  302.         // 商戶型別標識, 固定值  
  303.         keyValues.put("app_name", "mc");  
  304.         // 業務型別, 固定值  
  305.         keyValues.put("biz_type", "openservice");  
  306.         // 產品碼, 固定值  
  307.         keyValues.put("product_id", "APP_FAST_LOGIN");  
  308.         // 授權範圍, 固定值  
  309.         keyValues.put("scope", "kuaijie");  
  310.         // 商戶唯一標識,如:kkkkk091125  
  311.         keyValues.put("target_id", target_id);  
  312.         // 授權型別, 固定值  
  313.         keyValues.put("auth_type", "AUTHACCOUNT");  
  314.         // 簽名型別  
  315.         keyValues.put("sign_type", "RSA");  
  316.         return keyValues;  
  317.     }  
  318.     /** 
  319.      * 構造支付訂單引數列表 
  320.      * @param pid 
  321.      * @param app_id 
  322.      * @param target_id 
  323.      * @return 
  324.      */  
  325.     public static Map<String, String> buildOrderParamMap(String appId,String amount,String subject,String outTradeNo,String type) {  
  326.         Map<String, String> keyValues = new HashMap<String, String>();  
  327.         keyValues.put("app_id", appId);  
  328.         keyValues.put("biz_content", "{\"timeout_express\":\"30m\",\"product_code\":\"QUICK_MSECURITY_PAY\",\"total_amount\":\""+amount+"\",\"subject\":\""+subject+"\",\"out_trade_no\":\"" +outTradeNo +  "\"}");  
  329.         keyValues.put("charset", "utf-8");  
  330.         keyValues.put("method", "alipay.trade.app.pay");  
  331.         keyValues.put("sign_type", "RSA");  
  332.         //獲取域名   
  333.         SysParamServiceImpl systemParam = (SysParamServiceImpl)SpringContextHolder.getBean(SysParamServiceImpl.class);  
  334.         //String notifyUrl="http://180.96.11.10:8080/tourism";  
  335.         String notifyUrl=systemParam.getStringParamByKey("sys.domain");  
  336.         notifyUrl=notifyUrl+"/interface/alipay/"+type;  
  337.         keyValues.put("notify_url", notifyUrl);  
  338.         String timestamp = "";  
  339.         Date date=new Date();  
  340.         SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); //設定時間格式  
  341.         timestamp=sdf.format(date);  
  342.         keyValues.put("timestamp", timestamp);  
  343.         keyValues.put("version", "1.0");  
  344.         return keyValues;  
  345.     }  
  346.     /** 
  347.      * 構造支付訂單引數資訊 
  348.      *  
  349.      * @param map 
  350.      * 支付訂單引數 
  351.      * @return 
  352.      */  
  353.     public static String buildOrderParam(Map<String, String> map) {  
  354.         List<String> keys = new ArrayList<String>(map.keySet());  
  355.         StringBuilder sb = new StringBuilder();  
  356.         for (int i = 0; i < keys.size() - 1; i++) {  
  357.             String key = keys.get(i);  
  358.             String value = map.get(key);  
  359.             sb.append(buildKeyValue(key, value, true));  
  360.             sb.append("&");  
  361.         }  
  362.         String tailKey = keys.get(keys.size() - 1);  
  363.         String tailValue = map.get(tailKey);  
  364.         sb.append(buildKeyValue(tailKey, tailValue, true));  
  365.         return sb.toString();  
  366.     }  
  367.     /** 
  368.      * 拼接鍵值對 
  369.      *  
  370.      * @param key 
  371.      * @param value 
  372.      * @param isEncode 
  373.      * @return 
  374.      */  
  375.     private static String buildKeyValue(String key, String value, boolean isEncode) {  
  376.         StringBuilder sb = new StringBuilder();  
  377.         sb.append(key);  
  378.         sb.append("=");  
  379.         if (isEncode) {  
  380.             try {  
  381.                 sb.append(URLEncoder.encode(value, "UTF-8"));  
  382.             } catch (UnsupportedEncodingException e) {  
  383.                 sb.append(value);  
  384.             }  
  385.         } else {  
  386.             sb.append(value);  
  387.         }  
  388.         return sb.toString();  
  389.     }  
  390.     /** 
  391.      * 對支付引數資訊進行簽名 
  392.      *  
  393.      * @param map 
  394.      *            待簽名授權資訊 
  395.      *  
  396.      * @return 
  397.      */  
  398.     public static String getSign(Map<String, String> map, String rsaKey) {  
  399.         List<String> keys = new ArrayList<String>(map.keySet());  
  400.         // key排序  
  401.         Collections.sort(keys);  
  402.         StringBuilder authInfo = new StringBuilder();  
  403.         for (int i = 0; i < keys.size() - 1; i++) {  
  404.             String key = keys.get(i);  
  405.             String value = map.get(key).toString();  
  406.             authInfo.append(buildKeyValue(key, value, false));  
  407.             authInfo.append("&");  
  408.         }  
  409.         String tailKey = keys.get(keys.size() - 1);  
  410.         String tailValue =map.get(tailKey).toString();  
  411.         authInfo.append(buildKeyValue(tailKey, tailValue, false));  
  412.         String oriSign = SignUtils.sign(authInfo.toString(), rsaKey);  
  413.         String encodedSign = "";  
  414.         try {  
  415.             encodedSign = URLEncoder.encode(oriSign, "UTF-8");  
  416.         } catch (UnsupportedEncodingException e) {  
  417.             e.printStackTrace();  
  418.         }  
  419.         return "sign=" + encodedSign;  
  420.     }  
  421.     /** 
  422.      * 要求外部訂單號必須唯一。 
  423.      * @return 
  424.      */  
  425.     private static String getOutTradeNo() {  
  426.         SimpleDateFormat format = new SimpleDateFormat("MMddHHmmss", Locale.getDefault());  
  427.         Date date = new Date();  
  428.         String key = format.format(date);  
  429.         Random r = new Random();  
  430.         key = key + r.nextInt();  
  431.         key = key.substring(0, 15);  
  432.         return key;  
  433.     }  
  434. }</pre><br>  
  435. <span style="color:#ff0000">keyValues.put("notify_url", notifyUrl);這個和微信一樣,需要設定一個供支付寶支付後返回給我們的支付結果通知的請求,需自己定義,可帶引數,必須在外網下才能呼叫</span><br>  
  436. <p></p>  
  437. <p style="font-size:14px">最後根據我們定義的請求,來寫接收支付寶返回給我們的結果:</p>  
  438. <p style="font-size:14px"></p>  
  439. <pre name="code" class="java">@RequestMapping(value = "/interface/alipay/{type}", produces = "text/html;charset=UTF-8")  
  440.     @ResponseBody  
  441.     public String getzfbPayResult(HttpServletRequest request,HttpServletResponse response){  
  442.         try{  
  443.             String reqcontent = getRequestContent(request);  
  444.             interfaceLogger.info("################### ################# WeiXinPayResultInterface::alipay,request msg:"+reqcontent);  
  445.             //解析url  
  446.             //reqcontent=stringFarmat(reqcontent);   
  447.              try {  
  448.                  reqcontent= URLDecoder.decode(reqcontent, "UTF-8");  
  449.                     } catch (UnsupportedEncodingException e) {  
  450.                         e.printStackTrace();  
  451.                     }  
  452.             interfaceLogger.info("###########獲取引數解析url:[email protected]@@@@@@@@@@@@@@@@@@@@@@@@========="+reqcontent);  
  453.             //轉換為map  
  454.             Map<String, String> paramMap=stringtoArray(reqcontent);  
  455.             //獲取orderType  
  456.             String orderType=request.getRequestURI();  
  457.             int i=orderType.lastIndexOf("/");  
  458.             orderType=orderType.substring(i+1, orderType.length());   
  459.             //驗證簽名  
  460.             //支付寶公鑰  
  461.             String publicKey=Const.PUBLICKEY; //支付寶公鑰  
  462.              //字符集  
  463.             String charsetType="UTF-8";  
  464.             //驗證簽名  
  465.             boolean signVerified = AlipaySignature.rsaCheckV1(paramMap, publicKey, charsetType); //呼叫SDK驗證簽名  
  466.                 interfaceLogger.info("###########驗證簽名結果:[email protected]@@@@@@@@@@@@@@@@@@@@@@@@========="+signVerified);  
  467.                 if(signVerified){  
  468.                         //驗證金額訂單號  
  469.                     interfaceLogger.info("*****************驗證簽名成功:&&&&&&&&&&&&&&&&&&&&&&&&&&&&");  
  470.                     if(paramMap.containsKey("trade_status")){  
  471.                         String resultStr=paramMap.get("trade_status");  
  472.                         if(resultStr.equals("TRADE_SUCCESS")||resultStr.equals("TRADE_FINISHED")){  
  473.                             //支付成功後操作...  
  474.                     }  
  475.                     return "success";  
  476.                 }else{  
  477.                     interfaceLogger.info("&&&&&&&&&&&&&&&&驗證簽名失敗:***************************");  
  478.                    return "failure";      
  479.                 }  
  480.         }catch (Exception e){  
  481.             interfaceLogger.info("###########待簽名字串str:[email protected]@@@@@@@@@@@@@@@@@@@@@@@@=========",e);  
  482.             e.printStackTrace();  
  483.             return "failure";  
  484.         }  
  485.     }  
  486.   </pre>這一步需要商戶提供給我們的PUBLICKEY支付寶公鑰來進行證書驗證。<pre name="code" class="java">-------------------------------------------------------------------------------------------------------------------------------------</pre>這裡介紹了app上面的兩種支付方式。下面介紹下微信和支付寶的二維碼支付  
  487. <p></p>  
  488. <p style="font-size:14px">先看java後臺程式碼:</p>  
  489. <pre name="code" class="java"> /** 
  490.     *去支付介面(支付訂單) 
  491.     * @param request 
  492.     * @return 
  493.     * @throws Exception 
  494.     */  
  495.    @RequestMapping(value="/payOrder")  
  496.    public ModelAndView payOrder(HttpServletRequest request) throws Exception{  
  497.        ModelAndView mv = this.getModelAndView();  
  498.        PageData pd = new PageData();  
  499.        String type=request.getParameter("type");  
  500.        String orderId=request.getParameter("orderId");  
  501.        pd = this.getPageData();  
  502.        User user=SessionUtil.getUser(request);  
  503.       /* User user=new User(); 
  504.        user.setUserId("wang");*/  
  505.        String userId="";  
  506.        if(null!=user){  
  507.            userId=user.getUserId();  
  508.        }  
  509.        pd.put("userId", userId);  
  510.        pd=travelOrderService.getAllOrderDetail(pd);  
  511.        String orderNumber="";  
  512.        String amount="0.01";//測試金額  
  513.        String orderName="";  
  514.        if(pd.containsKey("orderName")){  
  515.            if(!StringUtils.isBlank(pd.get("orderName")))  
  516.                orderName=pd.get("orderName").toString();  
  517.        }  
  518.        if(pd.containsKey("orderNumber")){  
  519.            if(!StringUtils.isBlank(pd.get("orderNumber")))  
  520.                orderNumber=pd.get("orderNumber").toString();  
  521.        }  
  522.        //元轉成分   
  523.        String flag = sysparam.getStringParamByKey("system.flag");  
  524.        if(flag=="1"||flag.equals("1")){  
  525.            //type為0,3,4時取amount,type為1,2時取js_amount  
  526.            if(type.equals("0")||type.equals("3")||type.equals("4")){  
  527.                if(pd.containsKey("amount")){  
  528.                    if(!StringUtils.isBlank(pd.get("amount")))  
  529.                        amount=pd.get("amount").toString();  
  530.                }  
  531.            }else if(type.equals("1")||type.equals("2")){  
  532.                if(pd.containsKey("js_amount")){  
  533.                    if(!StringUtils.isBlank(pd.get("js_amount")))  
  534.                        amount=pd.get("js_amount").toString();  
  535.                }  
  536.            }  
  537.        }  
  538.        if (logger.isInfoEnabled())  
  539.        {  
  540.            logger.info("PayController:payOrder:amount:[" + amount + "]==>");  
  541.        }  
  542.        //以分為單位  
  543.        amount=floatToString(amount);  
  544.        //獲取生成微信二維碼的url和prepayId  
  545.        Map<String, Object> map=weChatPay(orderNumber,amount,type);  
  546.        String imageUrl="";  
  547.     f(map.containsKey("urlCode")){  
  548.            if(!StringUtils.isBlank(map.get("urlCode"))){  
  549.                imageUrl=map.get("urlCode").toString();  
  550.            }  
  551.        }  
  552.        //獲取生成支付寶二維碼的url(金額要改成元!)  
  553.        amount=stringToFloat(amount);  
  554.        String aliPayCodeUrl=AliPayUtil.aliPay(orderNumber, orderName, amount, type);  
  555.        map.put("type", type);  
  556.        map.put("orderNumber", orderNumber);  
  557.        map.put("aliPayCodeUrl", aliPayCodeUrl);  
  558.        //將二維碼入庫  
  559.        travelOrderService.updateOrderInfoByType(map);  
  560.        //訂單編號  
  561.        pd.put("type", type);  
  562.        pd.put("orderId", orderId);  
  563.        mv.addObject("pd", pd);  
  564.        mv.addObject("productId", orderNumber);  
  565.        mv.addObject("imageUrl", imageUrl);  
  566.        mv.addObject("aliPayCodeUrl", aliPayCodeUrl);  
  567.        mv.setViewName("/pay/pay-4");  
  568.        return mv;  
  569.    }</pre><br>  
  570. <p></p>  
  571. <p style="font-size:14px">這裡我把微信和支付寶支付前生成的二維碼寫在同一個方法裡,<span style="font-size:14px">imageURL是微信二維碼url,aliPayCodeUrl是支付寶的二維碼</span>,和app支付不同的是,掃碼支付需要事先生成二維碼,並且二維碼需要入庫,並帶到頁面去用於顯示(下面會講)</p>  
  572. <p style="font-size:14px">微信下單並生成urlCode</p>  
  573. <p style="font-size:14px"></p>  
  574. <pre name="code" class="java">/** 
  575.     * 微信下單並返回生成二維碼的url 
  576.     * 〈一句話功能簡述〉 
  577.     * 〈功能詳細描述〉 
  578.     * @param orderNo 
  579.     * @param TotalMoney 
  580.     * @param orderType 
  581.     * @return 
  582.     * @throws Exception String 
  583.     * 如果有違例,請使用@exception/throws [違例型別]   [違例說明:異常的註釋必須說明該異常的含義及什麼條件下丟擲該  
  584.     * @see          [類、類#方法、類#成員] 
  585.     */  
  586.    @SuppressWarnings("rawtypes")  
  587. public Map<String, Object> weChatPay(String orderNo,String TotalMoney,String orderType) throws Exception{  
  588.        URL postUrl = new URL("https://api.mch.weixin.qq.com/pay/unifiedorder");  
  589.        HttpsURLConnection con = (HttpsURLConnection) postUrl.openConnection();//開啟連線  
  590.        con.setRequestMethod("POST");//post方式提交  
  591.        con.setDoOutput(true);//開啟讀寫屬性,預設均為false  
  592.        con.setDoInput(true);  
  593.        con.setUseCaches(false);//Post請求不能使用快取  
  594.        con.setInstanceFollowRedirects(true);  
  595.        DataOutputStream out = new DataOutputStream(con.getOutputStream());  
  596.        Map<String, Object> vo = new HashMap<String, Object>();  
  597.        //相關引數,可以定義一個Map做成動態的  
  598.        String appid = sysparam.getStringParamByKey("c.wxpay.appid");  
  599.        vo.put("appid", appid);  
  600.        vo.put("attach", "支付測試");  
  601.        vo.put("body", "PC端支付測試");  
  602.        String mchid = sysparam.getStringParamByKey("c.wxpay.partnerid");  
  603.        vo.put("mch_id",mchid);  
  604.        vo.put("nonce_str", RandomUtil.CreateRandom(32));  
  605.        /String notifyUrl=sysparam.getStringParamByKey("sys.pc.domain");  
  606.        if (logger.isInfoEnabled())  
  607.        {  
  608.            logger.info("微信支付回撥地址設定:notifyUrl:[" + notifyUrl + "]==>");  
  609.        }  
  610.        notifyUrl=notifyUrl+"/travelWeb/pay/getWeChatPayResult/"+orderType; //呼叫微信下發通知的介面  
  611.        vo.put("notify_url", notifyUrl);  
  612.        vo.put("out_trade_no", orderNo);  
  613.        vo.put("spbill_create_ip", "14.23.150.211");  
  614.        vo.put("total_fee", TotalMoney);  
  615.        vo.put("trade_type", "NATIVE");  
  616.        //轉成sortedMap,微信支付後臺生成sign需要排序過的map  
  617.        SortedMap<String,Object> sort=new TreeMap<String,Object>(vo);  
  618.        //String secrit = "YNKSK648KG70j1085YYolajdfYUI7865";  
  619.        String secrit =sysparam.getStringParamByKey("c.wxpay.secret");  
  620.        //生成sign  
  621.        String sign=WeixinSignUtil.createSign("UTF-8", sort,secrit);  
  622.        //把sign放入map中  
  623.        vo.put("sign", sign);  
  624.        org.dom4j.Document doc = DocumentHelper.createDocument();  
  625.        Element body = DocumentHelper.createElement("xml");  
  626.        XmlUtil.buildMap2xmlBody(body,vo);  
  627.        doc.add(body);  
  628.        //傳送請求  
  629.        out.writeUTF(doc.asXML());    
  630.        System.out.println(doc.asXML());  
  631.        out.flush();  
  632.        out.close();  
  633.        //接收資料    
  634.        BufferedReader reader = new BufferedReader(new InputStreamReader(con.getInputStream(), "UTF-8"));    
  635.        String line;    
  636.        StringBuffer responseText = new StringBuffer();    
  637.        while ((line = reader.readLine()) != null) {    
  638.            responseText.append(line).append("\r\n");    
  639.        }    
  640.        reader.close();  
  641.        con.disconnect();    
  642.        String requestXML = responseText.toString()+"";   
  643.        if (logger.isInfoEnabled())  
  644.        {  
  645.            logger.info("weChatPay:輸出引數列表:[" + requestXML + "]==>");  
  646.        }  
  647.        //xml轉換為map  
  648.        Map map = XMLUtil.doXMLParse(requestXML);    
  649.        String urlCode = (String) map.get("code_url");    
  650.        String prepayId = (String) map.get("prepay_id");   
  651.        //返回生成二維碼的url和預支付交易會話id  
  652.        Map<String, Object> resultMap=new HashMap<String, Object>();  
  653.        resultMap.put("urlCode", urlCode);  
  654.        resultMap.put("prepayId", prepayId);  
  655.        return resultMap;   
  656.    }</pre>  
  657. <p></p>  
  658. <p style="font-size:14px">支付寶生成urlCode</p>  
  659. <p style="font-size:14px"></p>  
  660. <pre name="code" class="java">/** 
  661.      *  
  662.      * 〈一句話功能簡述〉 
  663.      * 〈功能詳細描述〉 
  664.      * @param orderNo 
  665.      *         訂單號 
  666.      * @param orderTitle 
  667.      *         訂單標題 
  668.      * @param orderPrice 
  669.      *         訂單總額 
  670.      * 如果有違例,請使用@exception/throws [違例型別]   [違例說明:異常的註釋必須說明該異常的含義及什麼條件下丟擲該  
  671.      * @see          [類、類#方法、類#成員] 
  672.      */  
  673.     public static String aliPay(String orderNo, String orderTitle,  
  674.                               String orderPrice,String type)  
  675.     {  
  676.         String aliPayCodeUrl="";  
  677.        // 支付超時,定義為120分鐘  
  678.         String timeoutExpress = "120m";  
  679.     // 建立掃碼支付請求builder,設定請求引數  
  680.        String notifyUrl=sysParamService.getStringParamByKey("sys.pc.domain");  
  681.         notifyUrl=notifyUrl+"/travelWeb/pay/getAliPayResult/"+type; //呼叫支付寶下發通知的介面  
  682.         AlipayTradePrecreateRequestBuilder builder = new AlipayTradePrecreateRequestBuilder()  
  683.             .setSubject(orderTitle)  
  684.             .setTotalAmount(orderPrice)  
  685.             .setOutTradeNo(orderNo)  
  686.             .setStoreId(storeId)  
  687.             .setTimeoutExpress(timeoutExpress)  
  688.             .setNotifyUrl(notifyUrl);//支付寶伺服器主動通知商戶伺服器裡指定的頁面http路徑,根據需要設定  
  689.        AlipayF2FPrecreateResult result = tradeService.tradePrecreate(builder);  
  690.         switch (result.getTradeStatus())  
  691.         {  
  692.             case SUCCESS:  
  693.                 logger.info("支付寶預下單成功: )");  
  694.                 AlipayTradePrecreateResponse response = result.getResponse();  
  695.                 System.out.println("response.getQrCode()====="  
  696.                                    + response.getQrCode());  
  697.                 aliPayCodeUrl=response.getQrCode();  
  698.                 break;  
  699.             case FAILED:  
  700.                 logger.error("支付寶預下單失敗!!!");  
  701.                 break;  
  702.             case UNKNOWN:  
  703.                 logger.error("系統異常,預下單狀態未知!!!");  
  704.                 break;  
  705.             default:  
  706.                 logger.error("不支援的交易狀態,交易返回異常!!!");  
  707.                 break;  
  708.         }  
  709.         return aliPayCodeUrl;  
  710.     }</pre><br>  
  711. <br>  
  712. <p></p>  
  713. <p style="font-size:14px"><br>  
  714. </p>  
  715. <p style="font-size:14px"><br>  
  716. </p>  
  717. 這個方法和app一樣,只是多返回了一個urlcode(urlcode生成二維碼,這兩個方法必須都要配置回撥地址<br>  
  718. 下一步需要編寫生成兩種二維碼的方法:  
  719. <p style="font-size:14px"></p>  
  720. <pre name="code" class="java">/** 
  721.     * 生成支付寶二維碼 
  722.     * 〈一句話功能簡述〉 
  723.     * 〈功能詳細描述〉 
  724.     * @param response 
  725.     * @param req void 
  726.     * 如果有違例,請使用@exception/throws [違例型別]   [違例說明:異常的註釋必須說明該異常的含義及什麼條件下丟擲該  
  727.     * @see          [類、類#方法、類#成員] 
  728.     */  
  729.    @RequestMapping(value = "/createAliPayCode.do", method = RequestMethod.GET)  
  730.    public void createAliPayCode(HttpServletResponse response,HttpServletRequest req)  
  731.    {  
  732.        String imageUrl=req.getParameter("aliPayCodeUrl");  
  733.        //傳入二維碼連線,圖片高度和寬頻  
  734.        BufferedImage image=ImageUtil.createImage(imageUrl,1000,1000);  
  735.        try  
  736.        {  
  737.            ImageIO.write(image, "gif", response.getOutputStream());  
  738.        }  
  739.        catch (IOException e)  
  740.        {  
  741.            e.printStackTrace();  
  742.        }  
  743.    }  
  744.    /** 
  745.     * 微信支付二維碼  
  746.     * 〈一句話功能簡述〉 
  747.     * 〈功能詳細描述〉 
  748.     * @param response 
  749.     * @param req void 
  750.     * 如果有違例,請使用@exception/throws [違例型別]   [違例說明:異常的註釋必須說明該異常的含義及什麼條件下丟擲該  
  751.  * @throws IOException  
  752.  * @throws JDOMException  
  753.     * @see          [類、類#方法、類#成員] 
  754.     */  
  755.    @RequestMapping(value = "/createWXPayCode.do", method = RequestMethod.GET)  
  756.    public void createWXPayCode(HttpServletResponse response,HttpServletRequest req) throws IOException, JDOMException  
  757.    {  
  758.        String imageUrl=req.getParameter("imageUrl");  
  759.        //傳入二維碼連線,圖片高度和寬頻  
  760.        BufferedImage image=ImageUtil.createImage(imageUrl,1000,1000);  
  761.        try  
  762.        {  
  763.            ImageIO.write(image, "gif", response.getOutputStream());  
  764.        }  
  765.        catch (IOException e)  
  766.        {  
  767.            e.printStackTrace();  
  768.        }  
  769.    }</pre>  
  770. <p></p>  
  771. <p style="font-size:14px">這裡生成二維碼的方法是用谷歌zhongxin(需要匯入jar包)</p>  
  772. <p style="font-size:14px"></p>  
  773. <pre name="code" class="java"> /** 
  774.      * 生成二維碼圖片方法 
  775.      * 〈一句話功能簡述〉 
  776.      * 〈功能詳細描述〉 
  777.      * @param text 
  778.      * @return BufferedImage 
  779.      * 如果有違例,請使用@exception/throws [違例型別]   [違例說明:異常的註釋必須說明該異常的含義及什麼條件下丟擲該  
  780.      * @see          [類、類#方法、類#成員] 
  781.      */  
  782.     @SuppressWarnings({"rawtypes", "unchecked"})  
  783.     public static BufferedImage createImage(String text,int width2,int height2)  
  784.     {  
  785.         //二維碼的圖片格式   
  786. //        String format = "gif";  
  787.         Hashtable hints = new Hashtable();  
  788.         //內容所使用編碼   
  789.         hints.put(EncodeHintType.CHARACTER_SET, "utf-8");  
  790.         BufferedImage image = null;  
  791.         try  
  792.         {  
  793.             BitMatrix bitMatrix = new MultiFormatWriter().encode(text,  
  794.                 BarcodeFormat.QR_CODE, width2, height2, hints);  
  795.             int width = bitMatrix.getWidth();  
  796.             int height = bitMatrix.getHeight();  
  797.             image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);  
  798.             for (int x = 0; x < width; x++ )  
  799.             {  
  800.                 for (int y = 0; y < height; y++ )  
  801.                 {  
  802.                     image.setRGB(x, y,  
  803.                         bitMatrix.get(x, y) ? 0xFF000000 : 0xFFFFFFFF); //二維碼圖片為黑白兩色  
  804.                 }  
  805.             }  
  806.         }  
  807.         catch (WriterException e)  
  808.         {  
  809.             e.printStackTrace();  
  810.         }  
  811.         return image;  
  812.     }</pre><br>  
  813. <br>  
  814. <p></p>  
  815. <br>  
  816. 這裡吧之前傳來的兩個值在面裡呼叫即可(img標籤配置src)  
  817. <p style="font-size:14px"></p>  
  818. <pre name="code" class="html"><div class="boder-line">  
  819.     <img src="<%=basePath%>/XX/pay/createWXPayCode.do?imageUrl=${imageUrl}"/>  
  820. </div>  
  821. <div class="boder-line">  
  822.     <img src="<%=basePath%>/XX/pay/createAliPayCode.do?aliPayCodeUrl=${aliPayCodeUrl}"/>  
  823. </div></pre><br>  
  824. 這樣開啟掃碼支付頁面就會有生成好的支付寶和微信支付二維碼圖片  
  825. <p></p>  
  826. <p style="font-size:14px">然後發起支付,最後根據配置的回撥地址返回支付結果</p>  
  827. <p style="font-size:14px">微信如下:</p>  
  828. <p style="font-size:14px"></p>  
  829. <pre name="code" class="java"> /** 
  830.     * 微信支付結果回撥地址 
  831.     * 〈一句話功能簡述〉 
  832.     * 〈功能詳細描述〉 
  833.     * @param request 
  834.     * @param response 
  835.  * @throws Exception  
  836.     * @see          [類、類#方法、類#成員] 
  837.     */  
  838.    @SuppressWarnings({"unchecked", "rawtypes"})  
  839.    @RequestMapping(value = "/getWeChatPayResult/{orderType}", produces = "text/html;charset=UTF-8")  
  840.    @ResponseBody  
  841.    public void getWeixinPayResult(HttpServletRequest request,  
  842.                                     HttpServletResponse response)  
  843.        throws Exception  
  844.    {  
  845.     //讀取引數    
  846.        InputStream inputStream ;    
  847.        StringBuffer sb = new StringBuffer();    
  848.        inputStream = request.getInputStream();    
  849.        String s ;    
  850.        BufferedReader in = new BufferedReader(new InputStreamReader(inputStream, "UTF-8"));    
  851.        while ((s = in.readLine()) != null){    
  852.            sb.append(s);    
  853.        }    
  854.        in.close();    
  855.        inputStream.close();    
  856.        //解析xml成map    
  857.        Map<String, String> m = new HashMap<String, String>();    
  858.        m = XMLUtil.doXMLParse(sb.toString());    
  859.        //獲取orderType  
  860.        String orderType=request.getRequestURI();  
  861.        int i=orderType.lastIndexOf("/");  
  862.        orderType=orderType.substring(i+1, orderType.length());   
  863.        logger.info("[PayController][getWeChatPayResult]:orderType:"+orderType);  
  864.        //過濾空 設定 TreeMap    
  865.        SortedMap<Object,Object> packageParams = new TreeMap<Object,Object>();          
  866.        Iterator it = m.keySet().iterator();    
  867.        while (it.hasNext()) {    
  868.            String parameter = (String) it.next();    
  869.            String parameterValue = m.get(parameter);    
  870.            String v = "";    
  871.            if(null != parameterValue) {    
  872.                v = parameterValue.trim();    
  873.            }    
  874.            packageParams.put(parameter, v);    
  875.        }    
  876.     String key =sysparam.getStringParamByKey("c.wxpay.secret");  
  877.        logger.info("[PayController][getWeChatPayResult]:packageParams:"+packageParams.toString());  
  878.        //判斷簽名是否正確    
  879.        if(PayCommonUtil.isTenpaySign("UTF-8", packageParams,key)) {    
  880.           String resXml = "";    
  881.            if("SUCCESS".equals((String)packageParams.get("result_code"))){    
  882.                // 這裡是支付成功    
  883.                //////////執行自己的業務邏輯////////////////    
  884.                String mch_id = (String)packageParams.get("mch_id");    
  885.                String openid = (String)packageParams.get("openid");    
  886.                String is_subscribe = (String)packageParams.get("is_subscribe");    
  887.                String o