1.微信支付也有很多种方式,今天我们所介绍的是公众号支付这种方式。这也是本人第一次进行微信开发,可能有一些地方介绍不到位,请见谅。
2.开发前准备,我们先去https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=11_1将一些工具下载下来,其实也可以自己写,只不过会有些麻烦,我们可以直接下载来使用。选择
Java版本【微信支付】API对应的SDK和调用示例的工具下载。
3.生成超链接跳转到我们的充值页面。
4.前台调用js进行异步发起下单请求。
5.后台接收到前台发起的请求,获取前台发过来的信息,将信息进行封装进行统一下单,需要加入到Map的信息有:appid(公众账号ID)、mch_id(商户号)、body(商品描述)、device_info(设备号)、out_trade_no(商户订单号)、fee_type(标价币种)、total_fee(标价金额)、spbill_create_ip(终端IP)、notify_url(通知地址)、trade_type(交易类型)、openid(用户标识)、nonce_str(随机字符串);也可以加入其它的,可以参考:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1,其中的请求参数,其中必填是的属于必须的,其它的可以视情况而定。调用wxpay.unifiedOrder(map)生成订单。
6.获取订单中的appId、timeStamp、nonceStr、package(值为prepay_id:prepay_id就是指下单时生成的订单id)、signType的值调用generateSignature(Map)生成签名,将生成的签名存放到五生成的订单中使用json的格式返回到前台中,此时前台可以获取后台传过来的值,接着前台调用callpay();在onBridgeReady()中,需要获取的值有:appId、timeStamp、nonceStr、package、signType、paySign(可能少一个都有可能回调不到微信输入密码页面),当出现微信密码输入页面的时候,我们就成功了一大半了。
7.支付成功后,接着开始处理后台回调,也就是notify_url里面的地址。注意:接收微信支付异步通知回调地址也是有要求:通知url必须为直接可访问的url,不能携带参数。同时,我们可以获取支付结果,我们们使用里面的参数生成一个本地签名,同时和结果集里面的签名比较,一样我们就可以进行我们的业务处理。处理完成后我们还要给通知一个回应,不然它会每隔一段时间它会再请求。
8.开发结束。
代码如下:
(1)、充值页面
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width,initial-scale=1,user-scalable=0">
<meta name="format-detection" content="telephone=no">
<title>微信充值</title>
</head>
<body>
<div class="vbox">
<div class="content">
<ul class="rechange-choose clearfix" id="rechange-choose">
<li class="active"><button class="btn btn-hollow-primary btn-w-per-100" data-value="30"><span class="money">30</span>元</button></li>
<li><button class="btn btn-hollow-primary btn-w-per-100" data-value="50"><span class="money">50</span>元</button></li>
<li><button class="btn btn-hollow-primary btn-w-per-100" data-value="100"><span class="money">100</span>元</button></li>
<li><button class="btn btn-hollow-primary btn-w-per-100" data-value="200"><span class="money">200</span>元</button></li>
</ul>
<div class="rechange-btn">
<button id="rechangeBtn" class="btn btn-primary btn-w-per-100">立即充值</button>
</div>
</div>
</div>
</body>
<script>
(function () {
var prepay_id,
paySign,
appId,
timeStamp,
nonceStr,
packageStr,
signType,
orderNo;
function onBridgeReady(){
WeixinJSBridge.invoke(
'getBrandWCPayRequest', {
"appId" : appId, //公众号名称,由商户传入
"timeStamp" : timeStamp, //时间戳数
"nonceStr" : nonceStr , //随机串
"package":"prepay_id="+prepay_id,
"signType" : 'MD5', //微信签名方式:
"paySign" : paySign //微信签名
},
function(res){
var last = JSON.stringify(res); //将JSON对象转化为JSON字符
if(res.err_msg === "get_brand_wcpay_request:ok" ) {
}
if (res.err_msg === "get_brand_wcpay_request:cancel") {
alert("交易取消");
}
if (res.err_msg === "get_brand_wcpay_request:fail") {
alert("支付失败");
}
}
);
}
function callpay(){
if (typeof WeixinJSBridge === undefined){
if( document.addEventListener ){
document.addEventListener('WeixinJSBridgeReady', onBridgeReady, false);
} else if (document.attachEvent){
document.attachEvent('WeixinJSBridgeReady', onBridgeReady);
document.attachEvent('onWeixinJSBridgeReady', onBridgeReady);
}
}else{
onBridgeReady();
}
}
/*点击支付*/
$('#rechangeBtn').on('click', function () {
payPublic();
});
function payPublic() {
var url = "支付处理地址";
Http.post(url, {
type : "add",
oppendId : '${oppendId!}',
total_fee:currentMoney
}, function (data) {
prepay_id = data.prepay_id;
paySign = data.sign;
appId = data.appid;
timeStamp = data.timeStamp;
nonceStr = data.nonce_str;
callpay();
}, function (err) {
});
}
}());
</script>
</html>
(2)、充值处理
protected ModelAndView handleRequestInternal(HttpServletRequest request, HttpServletResponse response) throws Exception { String type = request.getParameter("type"); String oppendId = request.getParameter("oppendId"); if (type != null) { if (type.equals("toadd")) { return new ModelAndView("weixin/pay/add").addObject("oppendId",oppendId); } else if (type.equals("add")) { /** * 公众号支付 下单 */ WXPayConfigImpl config =WXPayConfigImpl.getInstance(); WXPay wxpay = new WXPay(config); String totalFree=request.getParameter("total_fee"); String out_trade_no = UtilPramKey.getPriamKey("P");//订单号 double free =Double.valueOf(totalFree); free=free*100; totalFree=(int)free+""; HashMap<String, String> data = new HashMap<String, String>(); data.put("appid",config.getAppID()); data.put("mch_id",config.getMchID()); data.put("body", "产品描述"); data.put("device_info", "WEB"); data.put("out_trade_no", out_trade_no); data.put("fee_type", "CNY"); data.put("total_fee", totalFree); data.put("spbill_create_ip", "123.12.12.123"); data.put("notify_url", "异步通知地址"); data.put("trade_type", "JSAPI"); data.put("openid", oppendId); data.put("nonce_str", "随机数");//随机数 try { Map<String, String> r = wxpay.unifiedOrder(data); JSONObject js222 =new JSONObject(r); Date d =new Date(); if(r.get("return_code").equals("SUCCESS") && r.get("result_code").equals("SUCCESS")){ //业务处理模块点 }else{ //下单失败 } String timestr= d.getTime()/1000+""; r.put("timeStamp", timestr); String prepay_id=r.get("prepay_id"); Map<String, String> payMap = new HashMap<String, String>(); payMap.put("appId", config.getAppID()); payMap.put("timeStamp", timestr); payMap.put("nonceStr", r.get("nonce_str")); payMap.put("package", "prepay_id="+prepay_id); payMap.put("signType", "MD5"); String paySign = WXPayUtil.generateSignature(payMap, config.getKey()); r.put("sign", paySign); JSONObject js =new JSONObject(r); writeStringToResponse(js.toString(),response); } catch (Exception e) { e.printStackTrace(); } } } } return null; }
(3)、异步回调处理
protected ModelAndView handleRequestInternal(HttpServletRequest request, HttpServletResponse response) throws Exception { String re = getNotify(request); writeStringToResponse(re, response); return null; } public String getNotify(HttpServletRequest request) { String result = null; String inlength; String notifyXml = ""; try { while ((inlength = request.getReader().readLine()) != null) { notifyXml += inlength; } } catch (IOException e) {
//获取XML错误 }if (StringUtils.isEmpty(notifyXml)) {
//xml为空 } try { Map<String, String> map = WXPayUtil.xmlToMap(notifyXml); String appid=map.get("appid"); String bank_type=map.get("bank_type"); String cash_fee=map.get("cash_fee"); String device_info=map.get("device_info"); String fee_type=map.get("fee_type"); String is_subscribe=map.get("is_subscribe"); String mch_id=map.get("mch_id"); String nonce_str=map.get("nonce_str"); String openid=map.get("openid"); String out_trade_no=map.get("out_trade_no"); String result_code = map.get("result_code"); String return_code = map.get("return_code"); String sign=map.get("sign"); String time_end=map.get("time_end"); String total_fee=map.get("total_fee"); String trade_type=map.get("trade_type"); String transaction_id=map.get("transaction_id"); Map<String, String> date=new HashMap<>(); date.put("appid", appid); date.put("bank_type", bank_type); date.put("cash_fee", cash_fee); date.put("device_info", device_info); date.put("fee_type", fee_type); date.put("is_subscribe", is_subscribe); date.put("mch_id", mch_id); date.put("nonce_str", nonce_str); date.put("openid", openid); date.put("out_trade_no", out_trade_no); date.put("result_code", result_code); date.put("return_code", return_code); date.put("time_end", time_end); date.put("total_fee", total_fee); date.put("trade_type", trade_type); date.put("transaction_id", transaction_id); WXPayConfigImpl config = WXPayConfigImpl.getInstance(); String localSign=WXPayUtil.generateSignature(date, config.getKey()); if(localSign.equals(sign)){if (result_code.equals("SUCCESS") && return_code.equals("SUCCESS")) { //业务模块处理点 }else { //签名失败 } } catch (Exception e) { //转化XML错误 e.printStackTrace(); } //正确的结果要分上面几个不一样返回,不能这样只返回一个 result = "<xml>" + "<return_code><![CDATA[SUCCESS]]></return_code>" + "<return_msg><![CDATA[OK]]></return_msg>" + "</xml> "; return result;
}
(如若哪里存在漏洞请多多提出来,非常感谢)
