Android微信支付生成签名通用步骤讲解
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】
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】
//https://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; }
下一篇: 云计算时代下电子政务的机遇与挑战