忙了整整一天,累的要死,坚持更新。
以前都是根据第三方SDK来做支付,封装的确实不错,但是自己也是模模糊糊的,本文只是写下公众号的大概支付流程,所以并不会写太多的代码。
1、首先,假设我们已经配置好微信授权目录,,并且假设授权目录为:www.guohe.com
,那么在微信支付页面需填写下面的url
:
https://open.weixin.qq.com/connect/oauth2/authorize?appid=APPID&redirect_uri=REDIRECT_URI&response_type=code&scope=SCOPE&state=STATE#wechat_redirect
解释:
appid:公众号id
redirect_uri:后台支付接口和支付金额,例如:www.guohe.com/pay/money=100(必须将此链接进行encodeURI编码)
state:订单号码
2、后台处理支付功能代码:
在这里除了接收方法中传进来的金额、订单号码之外,还需要获取些微信支付需要的参数,例如公众号秘钥和公众号id等。
具体如下:
GZHID:微信公众号id
GZHSecret:微信公众号密钥id
SHHID:财付通商户号
SHHKEY:商户号对应的密钥
out_trade_no:商户订单号
money:金额
code:用户的code
nonce_str:随机数
spbill_create_ip:订单生成的机器 IP
notify_url:回调地址
通过code获取微信用户的openId和access_token,工具类:
/**
* 通过微信用户的code换取网页授权access_token
* @return
* @throws IOException
* @throws
*/
public List<Object> accessToken(String code) throws IOException {
List<Object> list = new ArrayList<Object>();
String url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid="
+ WeChat.HYGZHID + "&secret=" + WeChat.HYGZHSecret+ "&code=" + code + "&grant_type=authorization_code";
HttpClient client = new DefaultHttpClient();
HttpPost post = new HttpPost(url);
HttpResponse res = client.execute(post);
if (res.getStatusLine().getStatusCode() == HttpStatus.SC_OK) {
HttpEntity entity = res.getEntity();
String str = org.apache.http.util.EntityUtils.toString(entity, "utf-8");
ObjectMapper mapper=new com.fasterxml.jackson.databind.ObjectMapper.ObjectMapper();
Map<String,Object> jsonOb=mapper.readValue(str, Map.class);
list.add(jsonOb.get("access_token"));
list.add(jsonOb.get("openid"));
}
return list;
}
在这里生成预支付订单号的签名sign:
RequestHandler reqHandler = new RequestHandler(request, response);
reqHandler.init( GZHID, GZHSecret, SHHKEY);
String sign = reqHandler.createSign(packageParams);
生成官方稳定xml
数据:
String xml="<xml>"+
"<appid>"+ GZHID+"</appid>"+
"<mch_id>"+ SHHID+"</mch_id>"+
"<nonce_str>"+nonce_str+"</nonce_str>"+
"<sign>"+sign+"</sign>"+
"<body><![CDATA["+"费用"+"]]></body>"+
"<out_trade_no>"+out_trade_no+"</out_trade_no>"+
"<total_fee>"+finalmoney+"</total_fee>"+
"<spbill_create_ip>"+spbill_create_ip+"</spbill_create_ip>"+
"<notify_url>"+notify_url+"</notify_url>"+
"<trade_type>"+trade_type+"</trade_type>"+
"<openid>"+openid+"</openid>"+
"</xml>";
向微信官方统一接口发送数据:
String createOrderURL = "https://api.mch.weixin.qq.com/pay/unifiedorder";
String prepay_id="";
try {
prepay_id = GetWxOrderno.getPayNo(createOrderURL, xml);
if(prepay_id.equals("")){
mv.addObject("ErrorMsg", "支付错误");
mv.setViewName("error");
return mv;
}
} catch (Exception e) {
logger.error("统一支付接口获取预支付订单出错", e);
mv.setViewName("error");
return mv;
} else{
mv.addObject("SUCCESS", "支付成功");
mv.setViewName("wechat/pay");
}
3、回调地址方法的思路只是接收微信通知,并且返回就可以了,不然微信会一直发送通知的,微信发送的参数也是一段xml
格式的数据,是需要解析的,下面是解析工具类,做个记录:
/**
* description: 解析微信通知xml
*
* @param xml
* @return
*/
@SuppressWarnings({ "unused", "rawtypes", "unchecked" })
private static Map parseXmlToList(String xml) {
Map retMap = new HashMap();
try {
StringReader read = new StringReader(xml);
// 创建新的输入源SAX 解析器将使用 InputSource 对象来确定如何读取 XML 输入
InputSource source = new InputSource(read);
// 创建一个新的SAXBuilder
SAXBuilder sb = new org.jdom.input.SAXBuilder.SAXBuilder();
// 通过输入源构造一个Document
Document doc = (Document) sb.build(source);
Element root = doc.getRootElement();// 指向根节点
List<Element> es = root.getChildren();
if (es != null && es.size() != 0) {
for (Element element : es) {
retMap.put(element.getName(), element.getValue());
}
}
} catch (Exception e) {
e.printStackTrace();
}
return retMap;
}