Android集成微信支付功能
程序员文章站
2024-03-02 23:34:28
准备工作这里就不说了,包括签约和申请appid,附上微信开放平台app开发步骤,不懂的同学可以参考这里:
上面的步骤很详细,这里主要说下调起支付的注意事项。按照上面文...
准备工作这里就不说了,包括签约和申请appid,附上微信开放平台app开发步骤,不懂的同学可以参考这里:
上面的步骤很详细,这里主要说下调起支付的注意事项。按照上面文档中说的商户服务器生成支付订单,先调用统一下单api生成预付单,获取到prepay_id后将参数再次签名传输给app发起支付。
相关代码如下:
/** * 商户服务器生成支付订单,先调用统一下单api(详见第7节)生成预付单,获取到prepay_id后将参数再次签名传输给app发起支付。 */ //商品描述 string body = "iphone6s"; //随机字符串 string nonce_str = resourceutil.createrandomstring(32); //通知地址 string notify_url = "http://www.weixin.qq.com/wxpay/pay.php"; //商户订单号 string out_trade_no = resourceutil.generateouttradeno(32); //总金额(单位分) int total_fee = 1; string url = "https://api.mch.weixin.qq.com/pay/unifiedorder"; string sign = signutil.signbymd5("appid=" + constants.app_id + "&body=" + body + "&mch_id=" + constants.mch_id + "&nonce_str=" + nonce_str + "¬ify_url=" + notify_url + "&out_trade_no=" + out_trade_no + "&spbill_create_ip=127.0.0.1" + "&total_fee=" + total_fee + "&trade_type=app" + "&key=" + constants.key).touppercase(locale.getdefault()); //参数以xml格式传递 string entity = "<xml><appid>" + constants.app_id + "</appid><mch_id>" + constants.mch_id + "</mch_id><nonce_str>" + nonce_str +"</nonce_str><sign>" + sign + "</sign><body>" + body + "</body><out_trade_no>" + out_trade_no + "</out_trade_no><total_fee>" + total_fee + "</total_fee><spbill_create_ip>127.0.0.1</spbill_create_ip><notify_url>http://www.weixin.qq.com/wxpay/pay.php</notify_url><trade_type>app</trade_type></xml>"; log.d("entity", entity); paybutton.setenabled(false); toast.maketext(payactivity.this, "获取订单中...", toast.length_short).show(); byte[] buf = util.httppost(url, entity); if (buf != null && buf.length > 0) { string content = new string(buf); log.d("get server pay params:", content); orderresult orderresult = resourceutil.parsexml(new bytearrayinputstream(content.getbytes())); if (!textutils.equals(orderresult.getreturncode(), "success")) { toast.maketext(payactivity.this, orderresult.getreturnmsg(), toast.length_short).show(); return; } if (!textutils.equals(orderresult.getresultcode(), "success")) { toast.maketext(payactivity.this, orderresult.geterrordesc(), toast.length_short).show(); return; } //下单成功,调起支付 payreq request = new payreq(); request.appid = constants.app_id; request.partnerid = constants.mch_id; request.prepayid = orderresult.getprepayid(); request.packagevalue = "sign=wxpay"; request.noncestr = nonce_str; string timestamp = string.valueof(system.currenttimemillis() / 1000); request.timestamp = timestamp; request.sign = signutil.signbymd5("appid=" + constants.app_id + "&noncestr=" + nonce_str + "&package=sign=wxpay" + "&partnerid=" + constants.mch_id + "&prepayid=" + orderresult.getprepayid() + "×tamp=" + timestamp + "&key=" + constants.key).touppercase(locale.getdefault()); api.sendreq(request); paybutton.setenabled(true); } } });
相关参数说明在文档上都注明了,我这里面nonce_str和out_trade_no都是我随机生成的字符创,附上我的工具类,方便大家参考。
resourceutil.java
package com.xylpay.sdk.pay.uikit; import java.io.ioexception; import java.io.inputstream; import java.util.random; import org.xmlpull.v1.xmlpullparser; import org.xmlpull.v1.xmlpullparserexception; import com.xylpay.sdk.pay.bean.orderresult; import android.util.xml; public class resourceutil { /** * 随机生成字符串 * @param length 字符串的长度 * @return 随机字符串 */ public static string createrandomstring(int length) { string source = "0123456789abcdefghijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyz"; random random = new random(); stringbuilder builder = new stringbuilder(); for (int i = 0; i < length; i++) { int position = random.nextint(source.length()); builder.append(source.charat(position)); } return builder.tostring(); } public static string generateouttradeno(int n) { stringbuilder builder = new stringbuilder(); random random = new random(); for (int i = 0; i < n; i++) { builder.append(random.nextint(10)); } return builder.tostring(); } public static orderresult parsexml(inputstream is) { //pull解析xml数据 xmlpullparser parser = xml.newpullparser(); orderresult orderresult = null; try { parser.setinput(is, "utf-8"); int type = parser.geteventtype(); while(type != xmlpullparser.end_document) { switch (type) { case xmlpullparser.start_document: break; case xmlpullparser.start_tag: if (parser.getname().equals("xml")) { orderresult = new orderresult(); } else if (parser.getname().equals("return_code")) { orderresult.setreturncode(parser.nexttext()); } else if (parser.getname().equals("return_msg")) { orderresult.setreturnmsg(parser.nexttext()); } else if (parser.getname().equals("result_code")) { orderresult.setresultcode(parser.nexttext()); } else if (parser.getname().equals("err_code_des")) { orderresult.seterrordesc(parser.nexttext()); } else if (parser.getname().equals("prepay_id")) { orderresult.setprepayid(parser.nexttext()); } break; case xmlpullparser.end_tag: break; } type = parser.next(); } } catch(xmlpullparserexception e) { e.printstacktrace(); } catch (ioexception e) { e.printstacktrace(); } return orderresult; } }
其中关于sign的生成,参数的顺序一定要严格按照上面的顺序加上key进行md5加密,查看签名规范。
关于key的说明,这里的key是需要自己生成然后配置到微信开放平台的,参考商户支付密钥key的生成与设置进行配置,两边需要保持一致。另外,下单时,参数要以xml的格式来传递。
最后附上自己的签名算法:
signutil.java
package com.xylpay.sdk.pay.uikit; import java.security.messagedigest; import java.security.nosuchalgorithmexception; /** * created by jackie on 2016/2/15. * * md5加密 */ public class signutil { public static string signbymd5(string source) { byte[] bytes = null; try { messagedigest digest = messagedigest.getinstance("md5"); digest.update(source.getbytes()); //更新摘要 bytes = digest.digest(); //再通过执行诸如填充之类的最终操作完成哈希计算。在调用此方法之后,摘要被重置。 } catch (nosuchalgorithmexception e) { e.printstacktrace(); } stringbuilder builder = new stringbuilder(bytes.length * 2); for (byte b : bytes) { /** * 0xff默认是整形,一个byte跟0xff相与会先将那个byte转化成整形运算 */ if ((b & 0xff) < 0x10) { //如果为1位 前面补个0 builder.append("0"); } builder.append(integer.tohexstring(b & 0xff)); } return builder.tostring(); } }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。