使用场景:用于停车场收费系统,当车辆还在场内时,车主可以通过公众号里面的预缴费功能给车辆先缴停车费用,出场时可直接出场
用户使用流程:
- 车主打开微信公众号,进入预缴费功能界面
- 输入车牌号获取车辆停车信息
- 点击预交费生成停车缴费订单
- 点击支付跳转至微信支付界面
- 输入密码完成支付后点击完成界面关闭
业务实现流程:
- 根据车牌号获取车辆停车信息
- 根据车辆停车信息生成车辆缴费订单信息
- 根据车辆缴费订单信息拼接微信统一支付参数,并调用微信统一支付接口
- 支付完成后同步回调跳转至支付完成界面,异步回调跳转至支付成功接口
- 签名校验
- 保存支付成功账单信息并修改缴费订单支付状态
- 发送接收到回调应答
.
微信公众号支付开发者文档:https://pay.weixin.qq.com/wiki/doc/api/external/jsapi.php?chapter=7_1
一、统一下单:
接口地址:https://api.mch.weixin.qq.com/pay/unifiedorder
注:参数值用XML转义即可,CDATA标签用于说明数据不被XML解析器解析。
举例如下:
<xml>
<appid>wx2421b1c4370ec43b</appid>
<mch_id>10000100</mch_id>
<nonce_str>1add1a30ac87aa2db72f57a2375d8fec</nonce_str>
<body>JSAPI支付测试</body>
<out_trade_no>1415659990</out_trade_no>
<fee_type>GBP</fee_type>
<total_fee>1</total_fee>
<spbill_create_ip>14.23.150.211</spbill_create_ip>
<notify_url>http://wxpay.wxutil.com/pub_v2/pay/notify.v2.php</notify_url>
<trade_type>JSAPI</trade_type>
<openid>oUpF8uMuAJO_M2pxb1Q9zNjWeS6o</openid>
<attach>支付测试</attach>
<sign>0CB01533B8C1EF103065174F50BCA001</sign>
</xml>
返回示列:
<xml>
<return_code><![CDATA[SUCCESS]]></return_code>
<return_msg><![CDATA[OK]]></return_msg>
<appid><![CDATA[wx2421b1c4370ec43b]]></appid>
<mch_id><![CDATA[10000100]]></mch_id>
<nonce_str><![CDATA[IITRi8Iabbblz1Jc]]></nonce_str>
<sign><![CDATA[7921E432F65EB8ED0CE9755F0E86D72F]]></sign>
<result_code><![CDATA[SUCCESS]]></result_code>
<prepay_id><![CDATA[wx201411101639507cbf6ffd8b0779950874]]></prepay_id>
<trade_type><![CDATA[JSAPI]]></trade_type>
</xml>
微信签名生成规则:
- 将所有参数保存至集合M
- 将集合M内非空参数值的参数按照参数名ASCII码从小到大排序(字典序)
- 使用URL键值对的格式(即key1=value1&key2=value2…)拼接成字符串stringA
- 在stringA最后拼接上key得到stringSignTemp字符串
- 对stringSignTemp进行MD5运算
- 将得到的字符串所有字符转换为大写,得到sign值signValue
Java 中 TreeMap 的默认排序规则就是按照 key 的字典顺序来排序,正好与第二步想对应,所有可以直接将数据保存至 TreeMap 中
/**
* XML字符串转MAP
* @param xml
* @return
* @throws Exception
*/
public static Map<String, Object> xml2Map(String xml){
if (StringUtils.isEmpty(xml)) {
return null;
}
Map<String, Object> map = new HashMap<String, Object>();
try {
// 将xml格式的字符串转换成Document对象
Document doc = DocumentHelper.parseText(xml);
// 获取根节点
Element root = doc.getRootElement();
// 获取根节点下的所有元素
List children = root.elements();
// 循环所有子元素
if (children != null && children.size() > 0) {
for (int i = 0; i < children.size(); i++) {
Element child = (Element) children.get(i);
map.put(child.getName(), child.getTextTrim());
}
}
}catch (DocumentException e){
throw new RuntimeException(e);
}
return map;
}