您的位置:首页 > 移动开发 > 微信开发

【java】微信支付生成签名的过程

2016-10-26 18:10 302 查看
关于签名的算法,api提供的原文是:



1.签名算法

签名生成的通用步骤如下:

第一步,设所有发送或者接收到的数据为集合M,将集合M内非空参数值的参数按照参数名ASCII码从小到大排序(字典序),使用URL键值对的格式(即key1=value1&key2=value2…)拼接成字符串stringA。

特别注意以下重要规则:
 ◆ 参数名ASCII码从小到大排序(字典序);
 ◆ 如果参数的值为空不参与签名;
 ◆ 参数名区分大小写;
 ◆ 验证调用返回或微信主动通知签名时,传送的sign参数不参与签名,将生成的签名与该sign值作校验。
 ◆ 微信接口可能增加字段,验证签名时必须支持增加的扩展字段

第二步,在stringA最后拼接上key=(API密钥的值)得到stringSignTemp字符串,并对stringSignTemp进行MD5运算,再将得到的字符串所有字符转换为大写,得到sign值signValue。

举例:

假设传送的参数如下:

appid: wxd930ea5d5a258f4f

mch_id: 10000100  //微信支付商户号

device_info: 1000

body: test

nonce_str: ibuaiVcKdpRxkhJA  //随机字符串

第一步:对参数按照key=value的格式,并按照参数名ASCII字典序排序如下:

stringA="appid=wxd930ea5d5a258f4f&body=test&device_info=1000&mch_id=10000100&nonce_str=ibuaiVcKdpRxkhJA";

第二步:拼接API密钥:

stringSignTemp="stringA&key=192006250b4c09247ec02edce69f6a2d"

sign=MD5(stringSignTemp).toUpperCase()="9A0A8659F005D6984697E2CA0A9CF3B7"

网友的整理:



ok,根据这个我们来尝试一下看能够根据提供的参数得到这个结果【9A0A8659F005D6984697E2CA0A9CF3B7】!!!!

关于按照ASCII排序这里用到了(SortedMap),我的上一篇blog说过的,有兴趣自己去看看

关键代码Set es = parameters.entrySet();//所有参与传参的参数按照accsii排序(升序)

生成MD5的时候,需要统一编码,这里微信api要求是UTF-8

相关代码如下:

【MD5Util】

[java]
view plain
copy

print?

private static String byteArrayToHexString(byte b[]) {  
        StringBuffer resultSb = new StringBuffer();  
        for (int i = 0; i < b.length; i++)  
            resultSb.append(byteToHexString(b[i]));  
  
        return resultSb.toString();  
    }  
  
    private static String byteToHexString(byte b) {  
        int n = b;  
        if (n < 0)  
            n += 256;  
        int d1 = n / 16;  
        int d2 = n % 16;  
        return hexDigits[d1] + hexDigits[d2];  
    }  
  
    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;  
    }  
  
    private static final String hexDigits[] = { "0", "1", "2", "3", "4", "5",  
        "6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };  

【PayTest】

[java]
view plain
copy

print?

//http://mch.weixin.qq.com/wiki/doc/api/index.php?chapter=4_3  
    private static String Key = "192006250b4c09247ec02edce69f6a2d";  
  
    /** 
     * @param args 
     */  
    public static void main(String[] args) {  
        System.out.println(">>>模拟微信支付<<<");  
        System.out.println("==========华丽的分隔符==========");  
        //微信api提供的参数  
        String appid = "wxd930ea5d5a258f4f";  
        String mch_id = "10000100";   //微信支付商户号
        String device_info = "1000";  
        String body = "test";  
        String nonce_str = "ibuaiVcKdpRxkhJA";   //随机字符串
          
        SortedMap<Object,Object> parameters = new TreeMap<Object,Object>();  
        parameters.put("appid", appid);  
        parameters.put("mch_id", mch_id);  
        parameters.put("device_info", device_info);  
        parameters.put("body", body);  
        parameters.put("nonce_str", nonce_str);  
          
        String characterEncoding = "UTF-8";  
        String weixinApiSign = "9A0A8659F005D6984697E2CA0A9CF3B7";  
        System.out.println("微信的签名是:" + weixinApiSign);  
        String mySign = createSign(characterEncoding,parameters);  
        System.out.println("我     的签名是:"+mySign);  
          
        if(weixinApiSign.equals(mySign)){  
            System.out.println("恭喜你成功了~");  
        }else{  
            System.out.println("注定了你是个失败者~");  
        }  
          
        String userAgent = "Mozilla/5.0(iphone;CPU iphone OS 5_1_1 like Mac OS X) AppleWebKit/534.46(KHTML,like Geocko) Mobile/9B206 MicroMessenger/5.0";  
          
        char agent = userAgent.charAt(userAgent.indexOf("MicroMessenger")+15);  
          
        System.out.println("微信的版本号:"+new String(new char[]{agent}));  
    }  
  
    /** 
     * 微信支付签名算法sign 
     * @param characterEncoding 
     * @param parameters 
     * @return 
     */  
    @SuppressWarnings("unchecked")  
    public static String createSign(String characterEncoding,SortedMap<Object,Object> parameters){  
        StringBuffer sb = new StringBuffer();  
        Set es = parameters.entrySet();//所有参与传参的参数按照accsii排序(升序)  
        Iterator it = es.iterator();  
        while(it.hasNext()) {  
            Map.Entry entry = (Map.Entry)it.next();  
            String k = (String)entry.getKey();  
            Object v = entry.getValue();  
            if(null != v && !"".equals(v)   
                    && !"sign".equals(k) && !"key".equals(k)) {  
                sb.append(k + "=" + v + "&");  
            }  
        }  
        sb.append("key=" + Key);  
        String sign = MD5Util.MD5Encode(sb.toString(), characterEncoding).toUpperCase();  
        return sign;  
    }  

【最终结果】

[java]
view plain
copy

print?

>>>模拟微信支付<<<  
==========华丽的分隔符==========  
微信的签名是:9A0A8659F005D6984697E2CA0A9CF3B7  
我     的签名是:9A0A8659F005D6984697E2CA0A9CF3B7  
恭喜你成功了~  
微信的版本号:5  

后期用真正的商家APPId整合再试试~~good

转自:http://blog.csdn.net/xb12369/article/details/45716665
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: