微信掃碼支付

前些日子作的微信掃碼支付,如今都有點忘了怎麼作的了,趕忙記錄一下。。javascript

首先先看比較好的案例,幫了我好多好多,太感謝無私奉獻的人們了,我選擇的模式二開發html

1、場景:公司須要在網站上進行微信支付。

2、API:使用微信開放平臺的接入微信支付


-掃碼支付。微信支付開發者平臺連接java

3、實現:

      一是:按照微信掃碼支付規則生成二維碼信息.jquery

  二是:微信沒有提供生成二維碼圖片的接口。須要咱們本身把二維碼信息生成到二維碼圖片中。git

1、首先咱們要獲取微信後臺的必要的信息。web

public class Configure {
    // 每次本身Post數據給API的時候都要用這個key來對全部字段進行簽名,生成的簽名會放在Sign這個字段,API收到Post數據的時候也會用一樣的簽名算法對Post過來的數據進行簽名和驗證
    // 收到API的返回的時候也要用這個key來對返回的數據算下簽名,跟API的Sign數據進行比較,若是值不一致,有可能數據被第三方給篡改

    public static String key = "";
    
    //微信分配的公衆號ID(開通公衆號以後能夠獲取到)
    public static String appID = "";
    
    public static String appsecret = "";

    //微信支付分配的商戶號ID(開通公衆號的微信支付功能以後能夠獲取到)
    public static String mchID = "";

    //HTTPS證書的本地路徑
    public static String certLocalPath = "D:/wx/apiclient_cert.p12";

    //HTTPS證書密碼,默認密碼等於商戶號MCHID
    public static String certPassword = "";

    /**
     * 功能描述: 獲取客戶端IP
     *
     * @author yanzy  2016-1-18 下午08:19:45
     *
     * @param request
     * @return
     */
    public static String getClientIp(HttpServletRequest request) {
        String ip = request.getHeader("x-forwarded-for");
        if(ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
            ip = request.getHeader("Proxy-Client-IP");
        }
        if(ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
            ip = request.getHeader("WL-Proxy-Client-IP");
        }
        if(ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
            ip = request.getRemoteAddr();
        }
        return ip;
    }
    
    //隨機字符串
    public static String getNonceStr() {
        String currTime = PayCommonUtil.getCurrTime();  
        String strTime = currTime.substring(8, currTime.length());  
        String strRandom = PayCommonUtil.buildRandom(4) + "";
        String nonce_st = strTime + strRandom;  
        return nonce_st;
    }

    //1)掃碼支付API
    public static String PAY_API = "https://api.mch.weixin.qq.com/pay/unifiedorder";

    //2)退款API
    public static String REFUND_API = "https://api.mch.weixin.qq.com/secapi/pay/refund";

    //3)退款查詢API
    public static String REFUND_QUERY_API = "https://api.mch.weixin.qq.com/pay/refundquery";
    
    //4)查詢訂單狀態
    public static String PAY_QUERY_API = "https://api.mch.weixin.qq.com/pay/orderquery";
    
    public static String notify_url = BlockAttribute.host+"/payment/weixin/weixin_notify.jsp";

    public static String HttpsRequestClassName = "com.tencent.common.HttpsRequest";
 
}
還有各類工具類:HttpUtilajax

public class HttpUtil {
    private final static int CONNECT_TIMEOUT = 5000; // in milliseconds  
    private final static String DEFAULT_ENCODING = "UTF-8";  
      
    public static String postData(String urlStr, String data){  
        return postData(urlStr, data, null);  
    }  
      
    public static String postData(String urlStr, String data, String contentType){  
        BufferedReader reader = null;  
        try {  
            URL url = new URL(urlStr);  
            URLConnection conn = url.openConnection();  
            conn.setDoOutput(true);  
            conn.setConnectTimeout(CONNECT_TIMEOUT);  
            conn.setReadTimeout(CONNECT_TIMEOUT);  
            if(contentType != null)  
                conn.setRequestProperty("content-type", contentType);  
            OutputStreamWriter writer = new OutputStreamWriter(conn.getOutputStream(), DEFAULT_ENCODING);  
            if(data == null)  
                data = "";  
            writer.write(data);   
            writer.flush();  
            writer.close();    
 
            reader = new BufferedReader(new InputStreamReader(conn.getInputStream(), DEFAULT_ENCODING));  
            StringBuilder sb = new StringBuilder();  
            String line = null;  
            while ((line = reader.readLine()) != null) {  
                sb.append(line);  
                sb.append("\r\n");  
            }  
            return sb.toString();  
        } catch (IOException e) {
            LogUtil.writeLog("Error connecting to " + urlStr + ": " + e.getMessage());  
        } finally {  
            try {  
                if (reader != null)  
                    reader.close();  
            } catch (IOException e) {  
            }  
        }  
        return null;  
    }  
    
    /**
     * 自定義SSL雙向證書驗證
     *  
     * @param url
     * @param mchId
     * @param arrayToXml
     * @return
     * @throws Exception
     */  
    public static String clientCustomSSLCall(String url,String mchId, String arrayToXml) throws Exception {  
        String jsonStr = null;
        KeyStore keyStore = KeyStore.getInstance("PKCS12");
        // System.out.println("capath:=" + cAPath);  
        FileInputStream instream = new FileInputStream(new File(Configure.certLocalPath));  
        try {  
            keyStore.load(instream, mchId.toCharArray());  
        } finally {  
            instream.close();  
        }  
        // Trust own CA and all self-signed certs  
        SSLContext sslcontext = SSLContexts.custom().loadKeyMaterial(keyStore, mchId.toCharArray()).build();  
        // Allow TLSv1 protocol only  
        SSLConnectionSocketFactory sslsf = new SSLConnectionSocketFactory(sslcontext, new String[] { "TLSv1" }, null,SSLConnectionSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER);  
        CloseableHttpClient httpclient = HttpClients.custom().setSSLSocketFactory(sslsf).build();  
        try {  
            HttpPost httpPost = new HttpPost(url);  
            httpPost.setEntity(new StringEntity(arrayToXml, "UTF-8"));  
            CloseableHttpResponse response = httpclient.execute(httpPost);  
 
            jsonStr = EntityUtils.toString(response.getEntity(), "UTF-8");  
            response.close();  
        } finally {  
            httpclient.close();  
        }  
        return jsonStr;  
    }  
    
算法

MD5工具類json

public class MD5 {
    private final static String[] hexDigits = {"0", "1", "2", "3", "4", "5", "6", "7",
            "8", "9", "a", "b", "c", "d", "e", "f"};

    /**
     * 轉換字節數組爲16進制字串
     * @param b 字節數組
     * @return 16進制字串
     */
    public static String byteArrayToHexString(byte[] b) {
        StringBuilder resultSb = new StringBuilder();
        for (byte aB : b) {
            resultSb.append(byteToHexString(aB));
        }
        return resultSb.toString();
    }

    /**
     * 轉換byte到16進制
     * @param b 要轉換的byte
     * @return 16進制格式
     */
    private static String byteToHexString(byte b) {
        int n = b;
        if (n < 0) {
            n = 256 + n;
        }
        int d1 = n / 16;
        int d2 = n % 16;
        return hexDigits[d1] + hexDigits[d2];
    }

    /**
     * MD5編碼
     * @param origin 原始字符串
     * @return 通過MD5加密以後的結果
     */
    public static String MD5Encode(String origin, String charsetname) {  
        String resultString = null;  
        try {  
            resultString = new String(origin);  
            MessageDigest md = MessageDigest.getInstance("MD5");  
            if (charsetname == null || "".equals(charsetname))  
                resultString = byteArrayToHexString(md.digest(resultString  
                        .getBytes()));  
            else  
                resultString = byteArrayToHexString(md.digest(resultString  
                        .getBytes(charsetname)));  
        } catch (Exception exception) {  
        }  
        return resultString;  
    }  

}

api

PayCommonUtil 工具類

@SuppressWarnings({"rawtypes"})
public class PayCommonUtil {
    /**
     * 是否簽名正確,規則是:按參數名稱a-z排序,遇到空值的參數不參加簽名。
     * @return boolean
     */  
    public static boolean isTenpaySign(String characterEncoding, SortedMap<Object, Object> packageParams, String API_KEY) {  
        StringBuffer sb = new StringBuffer();  
        Set es = packageParams.entrySet();  
        Iterator it = es.iterator();  
        while(it.hasNext()) {  
            Map.Entry entry = (Map.Entry)it.next();  
            String k = (String)entry.getKey();  
            String v = (String)entry.getValue();  
            if(!"sign".equals(k) && null != v && !"".equals(v)) {  
                sb.append(k + "=" + v + "&");  
            }  
        }  
        sb.append("key=" + API_KEY);  
        //算出摘要  
        String mysign = MD5.MD5Encode(sb.toString(), characterEncoding).toLowerCase();  
        String tenpaySign = ((String)packageParams.get("sign")).toLowerCase();  
        return tenpaySign.equals(mysign);  
    }  
 
    /**
     * @Description:sign簽名
     * @param characterEncoding
     * @param parameters
     * @return
     */  
    public static String createSign(String characterEncoding, SortedMap<String, String> parameters, String API_KEY) {  
        StringBuffer sb = new StringBuffer();  
        Set es = parameters.entrySet();  
        Iterator it = es.iterator();  
        while (it.hasNext()) {  
            Map.Entry entry = (Map.Entry) it.next();  
            String k = (String) entry.getKey();  
            String v = (String) entry.getValue();  
            if (null != v && !"".equals(v) && !"sign".equals(k) && !"key".equals(k)) {  
                sb.append(k + "=" + v + "&");  
            }  
        }  
        sb.append("key=" + API_KEY);  
        String sign = MD5.MD5Encode(sb.toString(), characterEncoding).toUpperCase();  
        return sign;  
    }  
 
    /**
     * @author
     * @date 2016-4-22
     * @Description:將請求參數轉換爲xml格式的string
     * @param parameters
     * 請求參數
     * @return
     */  
    public static String getRequestXml(SortedMap<String, String> parameters) {  
        StringBuffer sb = new StringBuffer();  
        sb.append("<xml>");  
        Set es = parameters.entrySet();  
        Iterator it = es.iterator();  
        while (it.hasNext()) {  
            Map.Entry entry = (Map.Entry) it.next();  
            String k = (String) entry.getKey();  
            String v = (String) entry.getValue();  
            if ("attach".equalsIgnoreCase(k) || "body".equalsIgnoreCase(k) || "sign".equalsIgnoreCase(k)) {  
                sb.append("<" + k + ">" + "<![CDATA[" + v + "]]></" + k + ">");  
            } else {  
                sb.append("<" + k + ">" + v + "</" + k + ">");  
            }  
        }  
        sb.append("</xml>");  
        return sb.toString();  
    }  
 
    /**
     * 取出一個指定長度大小的隨機正整數.
     * @param length
     *  int 設定所取出隨機數的長度。length小於11
     * @return int 返回生成的隨機數。
     */  
    public static int buildRandom(int length) {  
        int num = 1;  
        double random = Math.random();  
        if (random < 0.1) {  
            random = random + 0.1;  
        }  
        for (int i = 0; i < length; i++) {  
            num = num * 10;  
        }  
        return (int) ((random * num));  
    }  
 
    /**
     * 獲取當前時間 yyyyMMddHHmmss
     * @return String
     */  
    public static String getCurrTime() {  
        Date now = new Date();  
        SimpleDateFormat outFormat = new SimpleDateFormat("yyyyMMddHHmmss");  
        String s = outFormat.format(now);  
        return s;  
    }  
 
    public static String getCurrTimeS() {
        Date now = new Date();
        SimpleDateFormat outFormat = new SimpleDateFormat("yyyy"+"年"+"MM"+"月"+"dd"+"日    "+"HH"+"時"+"mm"+"分"+"ss"+"秒");
        String s = outFormat.format(now);
        return s;
        
}
}  

XMLUtil  工具類

@SuppressWarnings({"unchecked"})
public class XMLUtil {
    /**
     * 解析xml,返回第一級元素鍵值對。若是第一級元素有子節點,則此節點的值是子節點的xml數據。
     * @param strxml
     * @return
     * @throws JDOMException
     * @throws IOException
     */  
    public static Map<String, Object> doXMLParse(String strxml) throws JDOMException, IOException {  
        strxml = strxml.replaceFirst("encoding=\".*\"", "encoding=\"UTF-8\"");  
 
        if(null == strxml || "".equals(strxml)) {  
            return null;  
        }  
          
        Map<String, Object> m = new HashMap<String, Object>();
          
        InputStream in = new ByteArrayInputStream(strxml.getBytes("UTF-8"));  
        SAXBuilder builder = new SAXBuilder();  
        Document doc = builder.build(in);  
        Element root = doc.getRootElement();  
        List<Element> list = root.getChildren();  
        Iterator<Element> it = list.iterator();  
        while(it.hasNext()) {  
            Element e = (Element) it.next();  
            String k = e.getName();  
            String v = "";  
            List<Element> children = e.getChildren();  
            if(children.isEmpty()) {  
                v = e.getTextNormalize();  
            } else {  
                v = XMLUtil.getChildrenText(children);  
            }  
              
            m.put(k, v);  
        }  
          
        //關閉流  
        in.close();  
          
        return m;  
    }  
      
    /**
     * 獲取子結點的xml
     * @param children
     * @return String
     */  
    public static String getChildrenText(List<Element> children) {  
        StringBuffer sb = new StringBuffer();  
        if(!children.isEmpty()) {  
             Iterator<Element> it = children.iterator();  
            while(it.hasNext()) {  
                Element e = (Element) it.next();  
                String name = e.getName();  
                String value = e.getTextNormalize();  
                List<Element> list = e.getChildren();  
                sb.append("<" + name + ">");  
                if(!list.isEmpty()) {  
                    sb.append(XMLUtil.getChildrenText(list));  
                }  
                sb.append(value);  
                sb.append("</" + name + ">");  
            }  
        }  
          
        return sb.toString();  
    }  
      

以上是須要用到的工具類,下面咱們就要開始調用了

 

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ page import="com.cn.service.face.payment.weixinpay.*"%>
<%@ page import="java.util.*"%>
<%@ page import="java.lang.*"%>
<%@ page import="com.cn.service.environment.*" %>
<%@ page import="com.cn.dto.entity.ordermanage.*"%>
<%@ page import="com.cn.service.face.order.commodity.*"%>
<%@ page import="com.cn.service.face.payment.weixinpay.*"%>
<%@ page import="com.cn.service.system.tools.*"%>
<%
        ///////////////////////微信掃碼支付接口      請求參數/////////////////////////////
        String orderno = request.getParameter("orderno");
        OrderManyPayFace orderManyPayFace=EnvironmentBean.getInstance().getBeanByBeanClass(OrderManyPayFace.class);
        UserOrder uo = orderManyPayFace.getUserOrder(orderno);
        double money=orderManyPayFace.getPayMoney(orderno);
        int transport=uo.getTransportCosts()!=null?uo.getTransportCosts().intValue():0;
        orderno=orderManyPayFace.getPayOrderno(orderno);
        money=money+transport;
        String txnAmt =Double.valueOf(money*100).intValue()+"";
        String body=uo.getOrderName();
    
        SortedMap<String,String> parameters = new TreeMap<String,String>();
        parameters.put("body", body);                          //商品描述              
        parameters.put("out_trade_no", orderno);            //商戶訂單號    
        parameters.put("total_fee", txnAmt);                 //交易金額,單位分,不要帶小數點          
        parameters.put("trade_type", "NATIVE");             //交易類型 ,
        parameters.put("product_id", orderno);              //商品ID
        
        WinxinpayServiceFace winxinpay=EnvironmentBean.getInstance().getBeanByBeanClass(WinxinpayServiceFace.class);
        String sHtmlText = winxinpay.trade_create_by_weixinpay(parameters,request);
        String qrcode=ZxingEncode.createQRCode(200, 200, request, sHtmlText);

%>
<script type="text/javascript" src="js/jquery-1.7.1.min.js"></script>
<script type="text/javascript" src="js/qrcode.js"></script>
<style>
*{ margin:0px; padding:0px;}
.wrapper{width:1000px; margin:20px auto;border:1px solid #ccc;}
.orderdetail{ background:#f2f2f2; height:60px; line-height:60px; color:#333333; padding-left:20px; padding-right:20px;}
.orderdetail span{ display:block;}
.ordernum{ float:left;}
.ordermoney{ float:right;}
.logo{ padding:20px;}
.pay_method{margin:20px; position:relative; height:400px;}
.pay_method ul{ position:absolute;left:0px;top:0px;background:#fff; z-index:1;}
.pay_method ul li{ float:left; list-style:none; border:1px solid #ccc;border-bottom:none; height:30px; line-height:30px; padding-left:20px; padding-right:20px;}
.num_pay{border:1px solid #ccc;position:absolute;left:0px;top:29px; width:960px; height:360px;}
.logowx{margin-top:20px;margin-left:20px;}
.erweima{ margin:0px auto; text-align:center;}
.erweima span{ display:block; margin-bottom:10px;}
</style>
<body>
<div class="wrapper">
    <div class="logo"><img src="images/pay/logo.png" width="61" height="26" /></div>
    <div class="orderdetail">
        <span class="ordernum">訂單編號: <%=orderno%></span>
        <span class="ordermoney">應付金額:&yen;<%=money%></span>
    </div>
    <div class="pay_method">
        <ul>
            <li>微信支付</li>
        </ul>
        <div style="clear:both;"></div>
        <div class="num_pay">
            <img src="images/pay/WePayLogo.png" width="120" height="32" class="logowx" />
            <div class="erweima">
               <!--  <span id="qrcode"></span> -->
                <img src="<%=qrcode%>"/>
                <span><img src="images/pay/des.png" width="180" height="60"/></span>
            </div>
        </div>
    </div>
</div>
</body>
<script type="text/javascript">
    //這是查詢訂單
    var t1;
    var sum=0;
    var out_trade_no="<%=orderno%>";
    var paymentId="<%=uo.getPaymentId()%>";
    t1=setInterval("ajaxstatus()", 5000);
    function ajaxstatus() {
        sum++;
        if(sum>600){ window.clearInterval(t1);return false;}
        if(sum>180){
            m=sum % 10;
            if(m!=0){return false;}
        }
        if (out_trade_no != null) {
            $.ajax({
             type:"post",
             url:"/PaymentFgAction!weixinQuery.action",
             dataType:"json",
             data: {"orderno":out_trade_no,"paymentId":paymentId},
             success: function (data) {
                 if (data== "1") {
                     window.location.href = "http://www.hiersun-ido.com/account/listUserOrder.html";
                     //<a href="http://www.hiersun-ido.com/account/listUserOrder.html">返回我的中心</a>
                 }
             },
                 error: function(XMLHttpRequest, textStatus, errorThrown) {
                 alert("請求訂單狀態出錯");
             }
             });
         }
     }
</script>

 

/**
     * 構造微信掃碼支付跳轉接口
     * @param sParaTemp 請求參數集合
     * @return 表單提交HTML信息
     * @throws IOException
     * @throws JDOMException
     */
    public static String trade_create_by_weixinpay(SortedMap<String, String> parameters,HttpServletRequest request) throws JDOMException, IOException {
        //增長基本配置
        parameters.put("appid", Configure.appID);
        parameters.put("mch_id", Configure.mchID);
        parameters.put("nonce_str",Configure.getNonceStr());
        parameters.put("spbill_create_ip", Configure.getClientIp(request));
        parameters.put("notify_url", Configure.notify_url);
        String sign = PayCommonUtil.createSign("UTF-8", parameters,Configure.key);  
        parameters.put("sign", sign);
        String requestXML = PayCommonUtil.getRequestXml(parameters);  
        String resXml = HttpUtil.postData(Configure.PAY_API, requestXML);
        Map map = XMLUtil.doXMLParse(resXml);
        String urlCode = (String) map.get("code_url");
        return urlCode;
    }

/**
     * 微信掃碼支付交易狀態查詢
     * @throws IOException
     * @throws JDOMException
     */
    public String weixinQuery(String orderno) throws JDOMException, IOException {
        
        SortedMap<String,String> parameters = new TreeMap<String,String>();
        //查詢交易須要的參數
        parameters.put("appid", Configure.appID);
        parameters.put("mch_id", Configure.mchID);
        parameters.put("out_trade_no", orderno);            //商戶訂單號
        parameters.put("nonce_str",Configure.getNonceStr());
        String sign = PayCommonUtil.createSign("UTF-8", parameters,Configure.key);  
        parameters.put("sign", sign);
        String requestXML = PayCommonUtil.getRequestXml(parameters);  
        
        String resXml = HttpUtil.postData(Configure.PAY_QUERY_API, requestXML);
        Map map = XMLUtil.doXMLParse(resXml);
        String trade_state  = (String) map.get("trade_state");
        return trade_state ;
    }
    
    /**
     * 構造微信掃碼申請退款接口
     * @param sParaTemp 請求參數集合
     * @return 表單提交HTML信息
     * @throws Exception
     */
    public static String trade_create_by_weixinrefund(SortedMap<String, String> parameters)  {
        String result_code=null;
        //增長基本配置
        try {
        parameters.put("appid", Configure.appID);
        parameters.put("mch_id", Configure.mchID);
        parameters.put("nonce_str",Configure.getNonceStr());
        String sign = PayCommonUtil.createSign("UTF-8", parameters,Configure.key);  
        parameters.put("sign", sign);
        String requestXML = PayCommonUtil.getRequestXml(parameters);  
        String resXml;
            resXml = HttpUtil.clientCustomSSLCall(Configure.REFUND_API, Configure.mchID,requestXML);
        //String resXml = HttpUtil.postData(Configure.REFUND_API, requestXML);
        Map map = XMLUtil.doXMLParse(resXml);
        result_code = (String) map.get("result_code");
        String err_code_des=(String)map.get("err_code_des");
        String refund_id=(String)map.get("refund_id");
        if("SUCCESS".equals(result_code)){
            return refund_id;
        }else{
            return err_code_des;
        }
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
            return "申請退款失敗";
        }  
    }

生成二維碼的工具類

import java.io.*;
import java.util.*;
import java.awt.*;
import java.awt.image.*;
import javax.imageio.*;
import javax.servlet.http.*;
import com.google.zxing.*;
import com.google.zxing.common.*;
import com.cn.service.system.common.*;
import com.google.zxing.qrcode.decoder.*;

@SuppressWarnings({ "rawtypes","unchecked","unused"})
public class ZxingEncode {

    public ZxingEncode() {
        super();
    }
    
    /**
     * 生成二維碼,並返回二維碼圖片路徑,文件名自動生成
     * content:二維碼內容
     */
    public static String createQRCode(int width,int height,HttpServletRequest request,String content) {
        String attachPath="";
        try {
            attachPath=AttachFile.getImagePath(request,BlockAttribute.twoDimensionCode);
            /*String projectName=AttachFile.getProjectName(attachPath);
            
            String temp_attachPath=attachPath;
            String mob_attachPath=attachPath;
            attachPath=attachPath.replaceAll(projectName, "OhdManage.war");
            temp_attachPath=temp_attachPath.replaceAll(projectName, "OhdStudio.war");    同步文件夾:一個文件同時存儲在二個項目下
            mob_attachPath=mob_attachPath.replaceAll(projectName, "OhdMobile.war");      同步文件夾:一個文件同時存儲在二個項目下
            */
            attachPath=AttachFile.createAttachDir(attachPath);
            /*temp_attachPath=AttachFile.createAttachDir(temp_attachPath);
            mob_attachPath=AttachFile.createAttachDir(mob_attachPath);*/
            
            String fileName=AttachFile.createFileName("png");
            attachPath=attachPath+FileTool.getFileSystemSeparator()+fileName;
            /*temp_attachPath=temp_attachPath+FileTool.getFileSystemSeparator()+fileName;
            mob_attachPath=mob_attachPath+FileTool.getFileSystemSeparator()+fileName;*/
            
            ZxingEncode qrcodeEncoder=new ZxingEncode();
            qrcodeEncoder.encoderQRCode(content, attachPath,width,height);
            /*qrcodeEncoder.encoderQRCode(content, temp_attachPath,width,height);
            qrcodeEncoder.encoderQRCode(content, mob_attachPath,width,height);
            
            attachPath=attachPath.substring(attachPath.indexOf("userfiles"));*/
        }
        catch(Exception err) {
            err.printStackTrace();
        }
        return AttachFile.getWebAttachPath(attachPath);
    }
    
    /**
     * 生成二維碼,並返回二維碼圖片路徑,須要傳入文件名
     * content:二維碼內容
     */
    public static String createQRCode(int width,int height,HttpServletRequest request,String content,String fileName) {
        String attachPath="";
        try {
            attachPath=AttachFile.getImagePath(request,BlockAttribute.twoDimensionCode);
            /*String projectName=AttachFile.getProjectName(attachPath);
            
            String temp_attachPath=attachPath;
            String mob_attachPath=attachPath;
            attachPath=attachPath.replaceAll(projectName, "OhdManage.war");
            temp_attachPath=temp_attachPath.replaceAll(projectName, "OhdStudio.war");    同步文件夾:一個文件同時存儲在二個項目下
            mob_attachPath=mob_attachPath.replaceAll(projectName, "OhdMobile.war");      同步文件夾:一個文件同時存儲在二個項目下 */
            
            attachPath=AttachFile.createAttachDir(attachPath);
            /*temp_attachPath=AttachFile.createAttachDir(temp_attachPath);
            mob_attachPath=AttachFile.createAttachDir(mob_attachPath);*/
            
            fileName=fileName+".png";
            attachPath=attachPath+FileTool.getFileSystemSeparator()+fileName;
            /*temp_attachPath=temp_attachPath+FileTool.getFileSystemSeparator()+fileName;
            mob_attachPath=mob_attachPath+FileTool.getFileSystemSeparator()+fileName;*/
            
            ZxingEncode qrcodeEncoder=new ZxingEncode();
            qrcodeEncoder.encoderQRCode(content, attachPath,width,height);
            /*qrcodeEncoder.encoderQRCode(content, temp_attachPath,width,height);
            qrcodeEncoder.encoderQRCode(content, mob_attachPath,width,height);
            
            attachPath=attachPath.substring(attachPath.indexOf("userfiles"));*/
        }
        catch(Exception err) {
            err.printStackTrace();
        }
        return AttachFile.getWebAttachPath(attachPath);
    }
    
    /**  
     * 生成二維碼(QRCode)圖片  
     * 用微信掃描GBK編碼的中文二維碼時出現亂碼,用UTF-8編碼時微信可正常識別 。
     * 而且MultiFormatWriter.encode()時若傳入hints參數來指定UTF-8編碼中文時,微信壓根就不識別所生成的二維碼。
     * 因此這裏使用的是這種方式new String(content.getBytes("UTF-8"), "ISO-8859-1")。
     * 生成的二維碼圖片默認背景爲白色,前景爲黑色,可是在加入logo圖像後會致使logo也變爲黑白色,至因而什麼緣由尚未仔細去讀它的源碼 。
     * 因此這裏對其第一個參數黑色將ZXing默認的前景色0xFF000000稍微改了一下0xFF000001,最終效果也是白色背景黑色前景的二維碼,且logo顏色保持原有。
     */
    public void encoderQRCode(String content,String imgPath,int width,int height) {
        try {
             MultiFormatWriter multiFormatWriter = new MultiFormatWriter();
             Map hints = new HashMap<EncodeHintType, String>();
             hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.M);  
             //hints.put(EncodeHintType.CHARACTER_SET, "UTF-8");
             BitMatrix bitMatrix = multiFormatWriter.encode(new String(content.getBytes("UTF-8"),"ISO-8859-1"), BarcodeFormat.QR_CODE, width, height,hints);
             File file = new File(imgPath);
             MatrixToImageWriter.writeToFile(bitMatrix, "png", file);
        }
        catch(Exception err) {
            err.printStackTrace();
        }
    }
    
    /**
     * 暫不支持加入頭象
     * 爲其二維碼添加頭象
     * 1)生成二維碼的糾錯級別建議採用最高等級H,這樣能夠增長二維碼的正確識別能力(我測試過,不設置級別時,二維碼工具沒法讀取生成的二維碼圖片)
     * 2)頭像大小最好不要超過二維碼自己大小的1/5,並且只能放在正中間部位,這是因爲二維碼自己結構形成的(你就把它理解成圖片水印吧)
     * 3)在仿照騰訊微信在二維碼四周增長裝飾框,那麼必定要在裝飾框和二維碼之間留出白邊,這是爲了二維碼可被識別
     */
    public void overlapImage(String imagePath, String logoPath) throws IOException {
        BufferedImage image = ImageIO.read(new File(imagePath));  
        int logoWidth = image.getWidth()/5;   //設置logo圖片寬度爲二維碼圖片的五分之一  
        int logoHeight = image.getHeight()/5; //設置logo圖片高度爲二維碼圖片的五分之一  
        int logoX = (image.getWidth()-logoWidth)/2;   //設置logo圖片的位置,這裏令其居中  
        int logoY = (image.getHeight()-logoHeight)/2; //設置logo圖片的位置,這裏令其居中  
        Graphics2D graphics = image.createGraphics();  
        graphics.drawImage(ImageIO.read(new File(logoPath)), logoX, logoY, logoWidth, logoHeight, null);  
        graphics.dispose();  
        ImageIO.write(image, imagePath.substring(imagePath.lastIndexOf(".") + 1), new File(imagePath));
    }
    
    private static final class MatrixToImageWriter {
        
        private static final int BLACK = 0xFF000000;
        private static final int WHITE = 0xFFFFFFFF;

        private MatrixToImageWriter() {
        }

        public static BufferedImage toBufferedImage(BitMatrix matrix) {
            int width = matrix.getWidth();
            int height = matrix.getHeight();
            BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
            for (int x = 0; x < width; x++) {
                for (int y = 0; y < height; y++) {
                    image.setRGB(x, y, matrix.get(x, y) ? BLACK : WHITE);
                }
            }
            return image;
        }

        public static void writeToFile(BitMatrix matrix, String format,File file) throws IOException {
            BufferedImage image = toBufferedImage(matrix);
            if (!ImageIO.write(image, format, file)) {
                throw new IOException("Could not write an image of format "    + format + " to " + file);
            }
        }
        
        public static void writeToStream(BitMatrix matrix, String format,OutputStream stream) throws IOException {
            BufferedImage image = toBufferedImage(matrix);
            if (!ImageIO.write(image, format, stream)) {
                throw new IOException("Could not write an image of format " + format);
            }
        }
    }
    
    public static void main(String[] args) {
        String imgPath = "D:/Michael_QRCode.png";
        ZxingEncode qrcodeEncoder=new ZxingEncode();
        qrcodeEncoder.encoderQRCode("http://192.168.12.33:8080/fdsfsafj-fkdsfks343243.html", imgPath,120,120);
    }
}

附件的保存與刪除
附件只保存在OhdFileMage項目下,圖片二維碼的工具類

public class AttachFile {

    public AttachFile() {
    }
            
    /**
     * 保存附件,返回文件路徑,type=0保 存的是文件,type=1保存的是圖片
     * clsPath:文件的分類文件夾名
     * 附件只保存在OhdFileMage項目下
     */
    public static String saveAttach(byte[] content,String extenName,int type,HttpServletRequest request,String clsPath) throws Exception {
        String attachPath="";
        try {
            if(type==0) {
                attachPath=getAttachPath(request,clsPath);
            }
            else if(type==1) {
                attachPath=getImagePath(request,clsPath);
            }
            //String projectName=getProjectName(attachPath);
            /*String temp_attachPath=attachPath;
            String mob_attachPath=attachPath;*/
            //attachPath=attachPath.replaceAll(projectName, "OhdFileMage.war");
            /*temp_attachPath=temp_attachPath.replaceAll(projectName, "OhdStudio.war");    同步文件夾:一個文件同時存儲在二個項目下
            mob_attachPath=mob_attachPath.replaceAll(projectName, "OhdMobile.war");      同步文件夾:一個文件同時存儲在二個項目下 */
            
            attachPath=createAttachDir(attachPath);
            /*temp_attachPath=createAttachDir(temp_attachPath);
            mob_attachPath=createAttachDir(mob_attachPath);*/
            
            String fileName=createFileName(extenName);
            attachPath=attachPath+FileTool.getFileSystemSeparator()+fileName;
            /*temp_attachPath=temp_attachPath+FileTool.getFileSystemSeparator()+fileName;
            mob_attachPath=mob_attachPath+FileTool.getFileSystemSeparator()+fileName;*/
            
            FileTool.writeFile(attachPath, content);
            /*FileTool.writeFile(temp_attachPath, content);
            FileTool.writeFile(mob_attachPath, content);*/
        }
        catch(Exception err) {
            throw err;
        }
        return getWebAttachPath(attachPath);
    }
    
    /**
     * 設置圖片路徑
     */
    public static String getWebAttachPath(String attachPath) {
        return "IdoFileMage/"+attachPath.substring(attachPath.indexOf("userfiles"));
    }
    
    /**
     * 返回項目名稱
     */
    public static String getProjectName(String attachPath) throws Exception {
        String projectName=null;
        if(attachPath.indexOf("IdoManage.war")>=0) {
            projectName="IdoManage.war";
        }
        else if(attachPath.indexOf("IdoStudio.war")>=0) {
            projectName="IdoStudio.war";
        }
        else if(attachPath.indexOf("IdoMobile.war")>=0) {
            projectName="IdoMobile.war";
        }
        return projectName;
    }
    
    /**
     * 按年月日建立文件夾
     */
    public static String createAttachDir(String attachPath) throws Exception {
        attachPath=attachPath+FileTool.getFileSystemSeparator()+DateTool.getYear();
        createAttachPath(attachPath);
        attachPath=attachPath+FileTool.getFileSystemSeparator()+(DateTool.getMonth()+1);
        createAttachPath(attachPath);
        /*attachPath=attachPath+FileTool.getFileSystemSeparator()+DateTool.getDay();    //文件夾只須要到月,到日產生太多的文件夾。
        createAttachPath(attachPath);*/
        return attachPath;
    }
    
    /**
     * 刪除文件
     */
    public static boolean deleteAttach(String attachPath) throws Exception {
        boolean deleteAttach=false;
        if(FileTool.isFileExist(attachPath)) {
            FileTool.deleteFile(attachPath);
            deleteAttach=true;
        }
        return deleteAttach;
    }
    
    /**
     * 建立指定的文件夾
     */
    private static void createAttachPath(String attachPath) throws Exception {
        if(!FileTool.isFileExist(attachPath)) {
            FileTool.createDir(attachPath);
        }
    }
    
    /**
     * 建立文件名
     */
    public static String createFileName(String extenName) {
        String filename=DateTool.dateToTimeString()+getRandom()+"."+extenName;
        return filename;
    }
    
    /**
     * 獲取web項目路徑
     * 注:附件只存在OhdFileMage.war項目下
     */
    public static String getWebPath(HttpServletRequest request) throws Exception {
        String attachPath=request.getSession().getServletContext().getRealPath("/");
        String projectName=getProjectName(attachPath);
        attachPath=attachPath.replaceAll(projectName, "IdoFileMage.war");
        return attachPath;
    }
    
    /**
     * 獲取附件的保存路徑
     * clsPath:文件的分類文件夾名
     */
    public static String getAttachPath(HttpServletRequest request,String clsPath) throws Exception {
        String attachPath=getWebPath(request);
        attachPath=attachPath+FileTool.getFileSystemSeparator()+"userfiles"+FileTool.getFileSystemSeparator()+"files"+FileTool.getFileSystemSeparator()+clsPath;
        createAttachPath(attachPath);
        return attachPath;
    }
    
    /**
     * 獲取附件的保存路徑
     * clsPath:文件的分類文件夾名
     */
    public static String getImagePath(HttpServletRequest request,String clsPath) throws Exception {
        String attachPath=getWebPath(request);
        attachPath=attachPath+FileTool.getFileSystemSeparator()+"userfiles"+FileTool.getFileSystemSeparator()+"images"+FileTool.getFileSystemSeparator()+clsPath;
        createAttachPath(attachPath);
        return attachPath;
    }
    
    /**
     * 生成一個1到1000的隨機數
     */
    public static int getRandom() {
        Random rd = new Random();
        return (1+rd.nextInt(1000));
    }
}


支付成功後的回調接口

<%
/* *
 功能:微信服務器異步通知頁面
 日期:2017-05-16
 
 * */
%>
<%@ page language="java" contentType="text/html; charset=GBK" pageEncoding="GBK"%>
<%@ page import="java.text.*"%>
<%@ page import="java.util.*"%>
<%@ page import="java.io.*"%>
<%@ page import="java.lang.*"%>
<%@ page import="com.cn.service.error.*" %>
<%@ page import="com.cn.service.environment.*" %>
<%@ page import="com.cn.service.face.payment.weixinpay.util.*"%>
<%@ page import="com.cn.service.face.order.commodity.*"%>
<%@ page import="com.cn.dto.entity.ordermanage.*"%>
<%@ page import="com.cn.service.face.order.*"%>
<%
    String paymentno=null;
    String out_trade_no=null;
    try{
        //讀取參數  
        InputStream inputStream;  
        StringBuffer sb = new StringBuffer();  
        inputStream = request.getInputStream();  
        String s;  
        BufferedReader in = new BufferedReader(new InputStreamReader(inputStream, "UTF-8"));  
        while ((s = in.readLine()) != null){  
            sb.append(s);  
        }  
        in.close();
        inputStream.close();
          //解析xml成map  
        Map<String, Object> m = new HashMap<String, Object>();
        m = XMLUtil.doXMLParse(sb.toString());
        
      //過濾空 設置 TreeMap  
        SortedMap<Object,Object> packageParams = new TreeMap<Object,Object>();
        Iterator it = m.keySet().iterator();
        while (it.hasNext()) {
            String parameter = (String) it.next();
            String parameterValue =(String) m.get(parameter);
            String v = "";
            if(null != parameterValue) {
                v = parameterValue.trim();  
            }  
            packageParams.put(parameter, v);  
        }  
         // 帳號信息  
        String key = Configure.key; // key
      //判斷簽名是否正確  
      if(PayCommonUtil.isTenpaySign("UTF-8", packageParams,key)) {
          String resXml = "";
          if("SUCCESS".equals((String)packageParams.get("result_code"))) {
              //獲取微信支付的通知返回參數,可參考技術文檔中頁面跳轉異步通知參數列表(如下僅供參考)
              String mch_id = (String)packageParams.get("mch_id");  
              String is_subscribe = (String)packageParams.get("is_subscribe");  
              out_trade_no = (String)packageParams.get("out_trade_no");
              String total_fee = (String)packageParams.get("total_fee");
              String transaction_id = (String)packageParams.get("transaction_id");
              String time_end = (String)packageParams.get("time_end");
              
              OrderPayRefundLogFace orderPayRefundLogFace=EnvironmentBean.getInstance().getBeanByBeanClass(OrderPayRefundLogFace.class);
                orderPayRefundLogFace.saveOrderPayLog(out_trade_no, "微信", packageParams.toString(), "");
              
              //執行本身的業務邏輯
              UserOrderFace userOrderFace= EnvironmentBean.getInstance().getBeanByBeanClass(UserOrderFace.class);
              UserOrder uo = userOrderFace.findUserOrder(out_trade_no);
              String money=String.valueOf(Double.valueOf(uo.getMoney().doubleValue()*100).intValue());
              if(money.equals(total_fee.trim())) {
                    userOrderFace.flushOrderStateSequence(out_trade_no,transaction_id,time_end);
                      //out.println("支付成功");
                      //通知微信.異步確認成功.必寫.否則會一直通知後臺.八次以後就認爲交易失敗了.  
                      resXml = "<xml>" + "<return_code><![CDATA[SUCCESS]]></return_code>"  
                          + "<return_msg><![CDATA[OK]]></return_msg>" + "</xml> ";
              }
              else {
                  out.println("假通知");  
              }
          }
          else {  
              out.println("支付失敗,錯誤信息:" + packageParams.get("err_code"));  
              resXml = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>"  
                      + "<return_msg><![CDATA[報文爲空]]></return_msg>" + "</xml> ";  
          }  
          out.println(resXml);  
      }
      else{  
          out.println("通知簽名驗證失敗");
      }  
    }
    catch(Exception ex) {
        out.println("通知簽名驗證失敗");  
        ex.printStackTrace();
        try {
            OrderPayRefundLogFace orderPayRefundLogFace=EnvironmentBean.getInstance().getBeanByBeanClass(OrderPayRefundLogFace.class);
            orderPayRefundLogFace.saveOrderPayLog(out_trade_no, "微信", "", ExceptionOut.writeException(ex));
        }
        catch(Exception error) {
            error.printStackTrace();
        }
    }
%>

退款的

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%><%@ page import="java.util.*"%><%@ page import="java.lang.*"%><%@ page import="com.cn.service.error.*" %><%@ page import="com.cn.service.environment.*" %><%@ page import="com.cn.service.system.tools.*" %><%@ page import="com.cn.dto.entity.ordermanage.*"%><%@ page import="com.cn.service.face.order.commodity.*"%><%@ page import="com.cn.service.face.payment.weixinpay.*"%><%@ page import="com.cn.service.face.payment.weixinpay.util.*"%><%    String transaction_id="";   //微信流水號(訂單號)    try {        ///////////////////////微信掃碼支付接口      請求參數/////////////////////////////        String orderno = request.getParameter("orderno");        /* type:1-定金,type:2-尾款 */        String type=request.getParameter("type");                String txnAmt="";           //正式的金額        String serviceOrderno="";   //退款批次號        String refundMoney="";      //退款金額        int refundMoneyInt=0;                UserOrderFace userOrderFace=EnvironmentBean.getInstance().getBeanByBeanClass(UserOrderFace.class);        UserOrder uo = userOrderFace.findUserOrder(orderno);        UserAfterSaleFace userAfterSaleFace=EnvironmentBean.getInstance().getBeanByBeanClass(UserAfterSaleFace.class);        AfterSaleService afterSaleService=userAfterSaleFace.getAfterSaleServic(uo.getServiceOrderno());        refundMoneyInt=afterSaleService.getRefundMoney()!=null?afterSaleService.getRefundMoney().intValue()*100:afterSaleService.getMoney().intValue()*100;        refundMoney=refundMoneyInt+"";        serviceOrderno=CodeCreator.createCodeNum12(1)[0];        if("0".equals(uo.getPayType())) {      //全款支付退款            transaction_id=uo.getTradeNo();                   txnAmt =Double.valueOf(uo.getMoney().doubleValue()*100).intValue()+"";        }        else if("1".equals(uo.getPayType())) { //定金支付            OrderManyPayFace orderManyPayFace=EnvironmentBean.getInstance().getBeanByBeanClass(OrderManyPayFace.class);            OrderManyPay orderManyPay1=orderManyPayFace.getOrderManyPay(orderno+"1");            OrderManyPay orderManyPay2=orderManyPayFace.getOrderManyPay(orderno+"2");            if("1".equals(type.trim())) {        //定金退款                transaction_id=orderManyPay1.getTradeNo();                txnAmt=Double.valueOf(orderManyPay1.getPaidMoney().doubleValue()*100).intValue()+"";                orderno=orderManyPay1.getPayOrderno();                if(refundMoneyInt>(orderManyPay1.getPaidMoney().intValue()*100)) {                    refundMoney=orderManyPay1.getPaidMoney().intValue()*100+"";                }            }            else if("2".equals(type.trim())) {   //尾款退款                transaction_id=orderManyPay2.getTradeNo();                txnAmt=Double.valueOf(orderManyPay2.getPaidMoney().doubleValue()*100).intValue()+"";                orderno=orderManyPay2.getPayOrderno();                int toal=orderManyPay1.getPaidMoney().intValue()*100+orderManyPay2.getPaidMoney().intValue()*100;                if(refundMoneyInt>=toal) {                //退款金額不能多支付金額                    refundMoney=orderManyPay2.getPaidMoney().intValue()*100+"";                }                else if(refundMoneyInt<toal && refundMoneyInt>(orderManyPay1.getPaidMoney().intValue()*100)) {                    refundMoney=refundMoneyInt-(orderManyPay1.getPaidMoney().intValue()*100)+"";                }                else if(refundMoneyInt<=(orderManyPay1.getPaidMoney().intValue()*100)) {                    refundMoney="0";                }            }        }        SortedMap<String,String> parameters = new TreeMap<String,String>();        parameters.put("transaction_id", transaction_id);             //微信訂單號            parameters.put("out_refund_no", serviceOrderno);              //商戶退款單號         parameters.put("total_fee", txnAmt);                              //交易金額,單位分,不要帶小數點                  parameters.put("refund_fee", refundMoney);                    //退款金額        parameters.put("appid", Configure.appID);        parameters.put("mch_id", Configure.mchID);        parameters.put("nonce_str",Configure.getNonceStr());        parameters.put("refund_account","REFUND_SOURCE_RECHARGE_FUNDS");        String sign = PayCommonUtil.createSign("UTF-8", parameters,Configure.key);          parameters.put("sign", sign);        String requestXML = PayCommonUtil.getRequestXml(parameters);        String resXml = HttpUtil.clientCustomSSLCall(Configure.REFUND_API, Configure.mchID,requestXML);        Map map = XMLUtil.doXMLParse(resXml);                String result_code =(String)map.get("result_code");         String err_code_des=(String)map.get("err_code_des");        String refund_id=(String)map.get("refund_id");                OrderPayRefundLogFace orderPayRefundLogFace=EnvironmentBean.getInstance().getBeanByBeanClass(OrderPayRefundLogFace.class);        orderPayRefundLogFace.saveOrderRefundLog(transaction_id, "微信", parameters.toString()+resXml, "");                if("SUCCESS".equals(result_code)) {            userAfterSaleFace.flushOrderStateSequence(transaction_id,refund_id);            userOrderFace.saveBatchNoSingle(orderno, serviceOrderno);     //保存退款批次號            response.sendRedirect("/IdoManage/UserOtoOrderAction!queryUserOrder.action");        }        else{            out.println(err_code_des);        }    }    catch(Exception err) {        err.printStackTrace();        try {            OrderPayRefundLogFace orderPayRefundLogFace=EnvironmentBean.getInstance().getBeanByBeanClass(OrderPayRefundLogFace.class);            orderPayRefundLogFace.saveOrderRefundLog(transaction_id, "微信","", ExceptionOut.writeException(err));        }        catch(Exception error) {            error.printStackTrace();        }    }%>

相關文章
相關標籤/搜索