Android中微信支付的流程(从请求统一支付接口到真正调起微信支付)

世事洞明皆学问,人情练达即文章。这篇文章主要讲述Android中微信支付的流程(从请求统一支付接口到真正调起微信支付)相关的知识,希望能为你提供帮助。
在公司做一款电商类的软件,接入支付是必不可少的环节。继上一次集成支付宝以后,微信支付又开启了另一段痛苦的历程。由于以前没有做过微信支付,所以这次在做的过程中还是遇到很大的问题。而且,公司目前没有自己的后台,所有的接口都是外包来承接的,在遇到问题时,外包一般会说,这是封装好的,以前都没有问题。然后,你只能自己查找原因,废话不多说,简单记录一下集成微信的整个过程。
1.微信支付的签名问题(包括微信的分享)
虽然关于微信的签名是个老生常谈的问题了,但是在这里我还是想要简单的描述一下.首先,要得到一个签名,你得先有一个自己的应用(android版).这就需要你到微信的开放平台上申请一个帐号,然后认证你的开发资质(这一步是不是必须我不太清楚),最后创建一个应用,进行应用的审核(这里需要填写你的应用包名和签名,当然这个后期也是可以修改的.这里的签名你可以在androidstudio上先对你的module进行签名,然后可以在微信的网站上下一个查看签名的工具,安装到手机上,输入你应用的包名,就可以查看你应用的签名了.查看签名工具的下载地址:
https://open.weixin.qq.com/zh_CN/htmledition/res/dev/download/sdk/Gen_Signature_Android.apk).审核通过以后,你就可以拿到你的appid和你的appsecreat(应该是通过以后,当然也可能不是).然后,你就需要申请开通app支付的功能,这期间就需要你作为一个商户之类的一些认证啊之类的东西,都是比较繁琐的.这些完成以后,就可以准备进行支付了.
2.导入微信支付的sdk
参照这里: http://www.jianshu.com/p/c97639279d2e
3.关于微信支付的请求统一接口以及二次签名
在完成上述的步骤以后,发现在调起微信支付的时候,还是会出现闪退的情况.于是就猜测会不会是后台返回给我的参数有问题,在这里跟后台核实了appid,partnerid,appsecret等参数.最后无果,还是在后台的一句以前都没有问题下无疾而终.所以在这种情况下,只能自己向微信去请求数据来得到自己需要的数据了(其实我是不想这么做的,因为以前没有接触过微信支付,但是在远程后台懒得管的情况下,只能自己去验证了).
首先,还是查阅了微信的官方文档:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_1
我承认,虽然我也看了几遍文档,可是对于怎么去完成请求还是不太理解.于是通过查阅网上各种信息,现总结如下:

3.1 准备工作
在你请求微信的统一支付的接口时,有几个参数是必须的,微信的文档上已经罗列出来.
(1) appid应用的id例如:wxd678efh567hg6787
(2) mch_id商户的id例如:1230000109(申请开通支付功能关联的商户的id)
(3) nonce_str随机字符串例如:5K8264ILTKCH16CQ2502SI8ZNMTM67VS
生成随机字符串你需要写一个方法,例如:

//一个10000以内的随机整数,并进行MD5加密 private String getNonce_str() { return MD5Utils.getMd5(new Random().nextInt(10000) + ""); }

(4) body商品描述例如:”ceshishangpin”(不确定这里是不是涉及到转码的问题,所以用的是字母)
(5) out_trade_no订单号例如:20150806125346(自家平台生成的订单号)
(7) spbill_create_ip终端ip例如:123.12.12.123(生成订单时设备的ip地址,我测试用的本机ip)
(8) notify_url通知地址例如:http://www.weixin.qq.com/wxpay/pay.php(这个地址还是要后台给你的,测试的话随便填也行吧应该,可以试试,不能包含特殊字符)
(9) trady_type交易类型例如:APP
(10) sign签名例如:C380BEC2BFD727A4B6845133519F3AD6 (这是微信的第一次签名,在这里你又需要写到一个方法了,MD5Utils中是用的UTF-8的编码方式,请自行准备这个工具类)
publicString createSign(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 + "& "); } } //这个partnerkey是需要自己进行设置的,要登陆你的微信的商户帐号(注意是商户,不是开放平台帐号),然后到api什么接口安全之类的那去设置,然后获取到 sb.append("key=" + Constant.WEIXIN_PARTERKEY); Log.e("TAG", sb.toString()); String sign = MD5Utils.getMd5(sb.toString()).toUpperCase(); Log.e("TAG", "sign的值为" + sign); return sign; }

可能有人要问了,你这个集合是个什么东西呢?我就是在别人那抄过来的其实…..
//参数:开始生成签名(这个类把这些参数封装到了一起) Unifiedorder unifiedorder = new Unifiedorder(); final SortedMap< Object, Object> parameters = new TreeMap< Object, Object> (); parameters.put("appid", Constant.WEIXIN_APPID); unifiedorder.setAppid(Constant.WEIXIN_APPID); parameters.put("mch_id", Constant.WEIXIN_PARTERID); unifiedorder.setMch_id(Constant.WEIXIN_PARTERID); //上面提到的获取随机数的方法 final String nonce_str = getNonce_str(); parameters.put("nonce_str", nonce_str); unifiedorder.setNonce_str(nonce_str); parameters.put("body", "ceshiweixinqianming"); unifiedorder.setBody("ceshiweixinqianming"); //order_id就是订单号 parameters.put("out_trade_no", order_id); unifiedorder.setOut_trade_no(order_id); //总金额 parameters.put("total_fee", 1); unifiedorder.setTotal_fee("1"); //ip地址 parameters.put("spbill_create_ip", "123.123.123.123"); unifiedorder.setSpbill_create_ip("123.123.123.123"); //支付成功的回调地址 String notify_url = "http://www.baidu.com/xxxx"; parameters.put("notify_url", notify_url); unifiedorder.setNotify_url(notify_url); parameters.put("trade_type", "APP"); unifiedorder.setTrade_type("APP"); //这里就是用上面的方法生成的sign值了 String sign = createSign(parameters); unifiedorder.setSign(sign);

还是把这个封装的类贴出来吧,毕竟搬砖也是挺累的……
//封装请求微信支付参数的bean类 class Unifiedorder { private String appid; private String mch_id; private String nonce_str; private String sign; private String body; private String out_trade_no; private String total_fee; private String spbill_create_ip; private String time_start; private String notify_url; private String trade_type; public String getAppid() { return appid; }public void setAppid(String appid) { this.appid = appid; }public String getMch_id() { return mch_id; }public void setMch_id(String mch_id) { this.mch_id = mch_id; }public String getNonce_str() { return nonce_str; }public void setNonce_str(String nonce_str) { this.nonce_str = nonce_str; }public String getSign() { return sign; }public void setSign(String sign) { this.sign = sign; }public String getBody() { return body; }public void setBody(String body) { this.body = body; }public String getOut_trade_no() { return out_trade_no; }public void setOut_trade_no(String out_trade_no) { this.out_trade_no = out_trade_no; }public String getTotal_fee() { return total_fee; }public void setTotal_fee(String total_fee) { this.total_fee = total_fee; }public String getSpbill_create_ip() { return spbill_create_ip; }public void setSpbill_create_ip(String spbill_create_ip) { this.spbill_create_ip = spbill_create_ip; }public String getTime_start() { return time_start; }public void setTime_start(String time_start) { this.time_start = time_start; }public String getNotify_url() { return notify_url; }public void setNotify_url(String notify_url) { this.notify_url = notify_url; }public String getTrade_type() { return trade_type; }public void setTrade_type(String trade_type) { this.trade_type = trade_type; } }

好了,sign值也设置到bean类中了,下面要做的就是按照微信要求的格式把这个bean类中的信息传给他,这时候你又需要另外的一个方法了.
//这个方法中需要注意的是,你在这个方法中拼接的参数,要和上面你已经赋给bean类的参数相一致,不能多也不能少,不然会出现签名错误的//请求微信的统一支付接口时需要用到的字符串信息 String xmlInfo = xmlInfo(unifiedorder); //微信的统一支付接口的地址 String wxUrl = "https://api.mch.weixin.qq.com/pay/unifiedorder"; publicString xmlInfo(Unifiedorder unifiedorder) { if (unifiedorder != null) { StringBuffer bf = new StringBuffer(); bf.append("< xml> "); bf.append("< appid> < ![CDATA["); bf.append(unifiedorder.getAppid()); bf.append("]]> < /appid> "); bf.append("< body> < ![CDATA["); bf.append(unifiedorder.getBody()); bf.append("]]> < /body> "); bf.append("< mch_id> < ![CDATA["); bf.append(unifiedorder.getMch_id()); bf.append("]]> < /mch_id> "); bf.append("< nonce_str> < ![CDATA["); bf.append(unifiedorder.getNonce_str()); bf.append("]]> < /nonce_str> "); bf.append("< notify_url> < ![CDATA["); bf.append(unifiedorder.getNotify_url()); bf.append("]]> < /notify_url> "); bf.append("< out_trade_no> < ![CDATA["); bf.append(unifiedorder.getOut_trade_no()); bf.append("]]> < /out_trade_no> "); bf.append("< spbill_create_ip> < ![CDATA["); bf.append(unifiedorder.getSpbill_create_ip()); bf.append("]]> < /spbill_create_ip> "); bf.append("< total_fee> < ![CDATA["); bf.append(unifiedorder.getTotal_fee()); bf.append("]]> < /total_fee> "); bf.append("< trade_type> < ![CDATA["); bf.append(unifiedorder.getTrade_type()); bf.append("]]> < /trade_type> "); bf.append("< sign> < ![CDATA["); bf.append(unifiedorder.getSign()); bf.append("]]> < /sign> "); bf.append("< /xml> "); Log.e("TAG", bf.toString()); return bf.toString(); }return ""; }

好了,按照微信要求的格式准备好字符串了,接下来就是向微信的接口地址请求数据了,我用的是okhttputils
OkHttpUtils.postString().content(xmlInfo).url(wxUrl).build().execute(new StringCallback() { @Override public void onError(Call call, Exception e, int id) {}@Override public void onResponse(String response, int id) { //如果顺利的话,这里就可以获取到微信返回给我们的信息了,当然如果不顺利的话,检查一下前几步有没有错误吧.... Log.e("TAG", response); } }

微信返回给我们的是一个这样的字符串:
< xml> < return_code> < ![CDATA[SUCCESS]]> < /return_code> < return_msg> < ![CDATA[OK]]> < /return_msg> < appid> < ![CDATA[wxxxxxxxxxxxxx]]> < /appid> < mch_id> < ![CDATA[xxxxxxxx]]> < /mch_id> < nonce_str> < ![CDATA[hQBELjqvxjPAKK7b]]> < /nonce_str> < sign> < ![CDATA[A499F6DC94AAC2648ADA31FD3AB7B806]]> < /sign> < result_code> < ![CDATA[SUCCESS]]> < /result_code> < prepay_id> < ![CDATA[wx20161208182704b5416397040869790726]]> < /prepay_id> < trade_type> < ![CDATA[APP]]> < /trade_type> < /xml>

得到这个字符串以后,需要进行解析,并提取出来我们需要的信息,我是这么做的…
//用来接收服务器返回的prepay_id参数 String prepay_id = ""; String nonce_str = ""; //我到现在都不明白服务器返回给我这个值有毛用 String sign = ""; XmlPullParser parser = Xml.newPullParser(); StringReader stringReader = new StringReader(response); try { parser.setInput(stringReader); int eventType = parser.getEventType(); while (eventType != XmlPullParser.END_DOCUMENT) { String nodeName = parser.getName(); switch (eventType) { case XmlPullParser.START_TAG: if ("prepay_id".equals(nodeName)) prepay_id = parser.nextText(); else if ("nonce_str".equals(nodeName)) { nonce_str = parser.nextText(); } else if ("sign".equals(nodeName)) { sign = parser.nextText(); } break; } //这一行代码不能丢,我把这丢了,然后,死循环了... eventType = parser.next(); }} catch (Exception e) { e.printStackTrace(); } //关闭流..有用么? stringReader.close();

好了,到这是不是觉得我们需要的参数都已经得到了?其实我起初也是这么想的..然而..还是会出现闪退..于是乎..想到了前辈们说的二次签名,然后查了一下,最后抱着试一试的心态……
req.appId = Constant.WEIXIN_APPID; req.partnerId = Constant.WEIXIN_PARTERID; req.prepayId = prepay_id; req.packageValue = "https://www.songbingjia.com/android/Sign=WXPay"; req.nonceStr = nonce_str; //这是得到一个时间戳(除以1000转化成秒数) req.timeStamp = System.currentTimeMillis() / 1000 + ""; //这个集合是上面用到的那个集合,因为我是写在一起的,就直接clear了一下接着用了,下面的这些就是二次签名 parameters.clear(); parameters.put("appid", Constant.WEIXIN_APPID); parameters.put("partnerid", Constant.WEIXIN_PARTERID); parameters.put("prepayid", prepay_id); parameters.put("noncestr", nonce_str); parameters.put("timestamp", req.timeStamp); parameters.put("package", req.packageValue); //调用获得签名的方法,这里直接把服务器返回来的sign给覆盖了,所以我不是很明白服务器为什么返回这个sign值,然后调起支付,基本上就可以了(我的反正是可以了....) sign = createSign(parameters); Log.e("TAG", "timestamp=====" + req.timeStamp); req.sign = sign; // 在支付之前,如果应用没有注册到微信,应该先调用IWXMsg.registerApp将应用注册到微信 api.sendReq(req);

【Android中微信支付的流程(从请求统一支付接口到真正调起微信支付)】到这里,基本上我遇到的问题都解决完了,我觉得最大的问题还是因为没有后台的支持,需要自己对这些参数进行检验,而且从前没有进行过类似的工作.在此进行一下记录,希望对遇到同样问题的同学有所帮助.


















    推荐阅读