首先我们先看一个Android微信支付时遇到的一个错误提示:
onPayFinish, errCode = -1
当你参数签名都没有问题的时候,出现这个提示,请按照如下操作:
在你的项目测试android微信的组件(微信分享、微信支付等)的时候,一定要用你自己的keystore签名出来测试,如果用debug.keystore肯定是不成功的!
本文部分内容转自:http://blog.csdn.NET/baidu_17508977/article/details/44517283
1,在微信开放平台申请app_id,app_key我就不在这里叙述了,稍后我会把开发文档一并上传的,你也可以去微信开放平台自行查看(差不多一个周才会通过审核)
https://open.weixin.qq.com/cgi-bin/index?t=home/index&lang=zh_CN&token=44d65f9f1df0d4725b941e217a0bf769fbb51b3a
2,下面我就来说一说微信支付开发时需要注意的地方:
2.1 首先来看一下,微信支付的架构和流程图
3.2 再来看一下成功调起微信支付的界面
3下面开始讲解配置工程
3.1 这里必须要有wxapi这个包名,同时必须有WXPayEntryActivity这个类名,否则无法调起微信支付,(开发文档没有标注,废了好大周章)
3.2 支付成功通知:在WXPayEntryActivity的OnResp中处理,不能以微信返回的通知界面为准(我遇到的情况,网络不稳定的时候,微信返回界面提示支付失败,但是收到微信通知其实已经支付成功了),必须要去自己的服务器查询支付状态,这里微信建议用轮循机制去查询(最好听微信劝,O(∩_∩)O哈哈~)
- @Override
- public void onResp(BaseResp resp) {
- Log.d(TAG, "onPayFinish, errCode = " + resp.errCode);
-
- if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX) {
- AlertDialog.Builder builder = new AlertDialog.Builder(this);
- builder.setTitle(R.string.app_tip);
- builder.setMessage(getString(R.string.pay_result_callback_msg, resp.errStr +";code=" + String.valueOf(resp.errCode)));
- builder.show();
- }
- }
3.3 生成prepay_id,在服务器完成,由服务器去跟微信服务器交互,客户端不需要参与
-
-
-
-
-
-
- private String genProductArgs() {
- StringBuffer xml = new StringBuffer();
-
- try {
- String nonceStr = genNonceStr();
-
-
- xml.append("</xml>");
- List<NameValuePair> packageParams = new LinkedList<NameValuePair>();
- packageParams.add(new BasicNameValuePair("appid", Constants.APP_ID));
- packageParams.add(new BasicNameValuePair("body", "APP pay test"));
-
- packageParams.add(new BasicNameValuePair("mch_id", Constants.MCH_ID));
- packageParams.add(new BasicNameValuePair("nonce_str", nonceStr));
- packageParams.add(new BasicNameValuePair("notify_url", "http://121.40.35.3/test"));
- packageParams.add(new BasicNameValuePair("out_trade_no",genOutTradNo()));
- packageParams.add(new BasicNameValuePair("spbill_create_ip","127.0.0.1"));
- packageParams.add(new BasicNameValuePair("total_fee", "1"));
- packageParams.add(new BasicNameValuePair("trade_type", "APP"));
-
-
- String sign = genPackageSign(packageParams);
- packageParams.add(new BasicNameValuePair("sign", sign));
-
-
- String xmlstring =toXml(packageParams);
-
- return xmlstring;
-
- } catch (Exception e) {
- Log.e(TAG, "genProductArgs fail, ex = " + e.getMessage());
- return null;
- }
-
-
- }
3.4 获取二次签名sign
- private void genPayReq() {
-
- req.appId = Constants.APP_ID;
- req.partnerId = Constants.MCH_ID;
- req.prepayId = resultunifiedorder.get("prepay_id");
-
- req.packageValue = "Sign=WXPay";
- req.nonceStr = genNonceStr();
- req.timeStamp = String.valueOf(genTimeStamp());
-
-
- List<NameValuePair> signParams = new LinkedList<NameValuePair>();
- signParams.add(new BasicNameValuePair("appid", req.appId));
- signParams.add(new BasicNameValuePair("noncestr", req.nonceStr));
-
-
-
-
- signParams.add(new BasicNameValuePair("package", req.packageValue));
-
- signParams.add(new BasicNameValuePair("partnerid", req.partnerId));
- signParams.add(new BasicNameValuePair("prepayid", req.prepayId));
- signParams.add(new BasicNameValuePair("timestamp", req.timeStamp));
-
- req.sign = genAppSign(signParams);
-
- sb.append("sign\n"+req.sign+"\n\n");
-
- show.setText(sb.toString());
-
- Log.e("orion", signParams.toString());
-
- }
3.5 起调微信支付
- private void sendPayReq() {
- msgApi.registerApp(Constants.APP_ID);
- msgApi.sendReq(req);
- }
3.6 配置Manifest.xml,权限什么的按照文档的配置就行了
- <activity
- android:name=".PayActivity"
- android:label="@string/app_name"
- android:exported="true"
- android:launchMode="singleTop">
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
-
- <intent-filter>
- <action android:name="android.intent.action.VIEW"/>
- <category android:name="android.intent.category.DEFAULT"/>
- <data android:scheme="wxd930ea5d5a258f4f"/>
- </intent-filter>
- </activity>
4,支付通知接口和退款接口按照开发文档即可,这里不再赘述
5, 在你的项目测试微信支付的时候,一定要用你自己的keystore签名出来测试,如果用debug.keystore肯定是不成功的,切记,切记,不要闹乌龙!!!
好了,就写这么多吧,以此文来祭奠我和同事被腾讯坑死的那几天,如果有不懂的童鞋可以给我留言,或者QQ:1031012395联系我,大家可以去这个地址下载微信支付v3.0版本的开发文档和demo,以及一个成功的微信支付demo app,轮询事件大家可以参考我的下一篇blog,从别人那里转来的,写的挺详细的
http://download.csdn.Net/detail/baidu_17508977/8521101
补充问题:
微信支付v3 body中文无法支付问题
String nonceStr = genNonceStr();
xml.append("</xml>");// Yuebai Steam Car Wash Service
List<NameValuePair> packageParams = new LinkedList<NameValuePair>();
packageParams.add(new BasicNameValuePair("appid", Constants.APP_ID));
packageParams.add(new BasicNameValuePair("body", "月白洗车"));// 这个一改就无法支付
packageParams.add(new BasicNameValuePair("mch_id", Constants.MCH_ID));
packageParams.add(new BasicNameValuePair("nonce_str", nonceStr));
packageParams.add(new BasicNameValuePair("notify_url", HttpConstant.wxapi));
packageParams.add(new BasicNameValuePair("out_trade_no", orderid));
packageParams.add(new BasicNameValuePair("spbill_create_ip", "127.0.0.1"));
int a = (int) (Integer.parseInt(m) * 100);
packageParams.add(new BasicNameValuePair("total_fee", a + ""));
packageParams.add(new BasicNameValuePair("trade_type", "APP"));
String sign = genPackageSign(packageParams);
packageParams.add(new BasicNameValuePair("sign", sign));
String xmlstring = toXml(packageParams);
return new String(xmlstring.toString().getBytes(), "ISO8859-1");//这句加上就可以了吧xml转码下
另外说明的是,如果想增加参数,请先看看下面的内容。
签名算法
签名生成的通用步骤如下:
第一步,设所有发送或者接收到的数据为集合M,将集合M内非空参数值的参数按照参数名ASCII码从小到大排序(字典序),使用URL键值对的格式(即key1=value1&key2=value2…)拼接成字符串stringA。
特别注意以下重要规则:
- ◆ 参数名ASCII码从小到大排序(字典序);
- ◆ 如果参数的值为空不参与签名;
- ◆ 参数名区分大小写;
- ◆ 验证调用返回或微信主动通知签名时,传送的sign参数不参与签名,将生成的签名与该sign值作校验。
- ◆ 微信接口可能增加字段,验证签名时必须支持增加的扩展字段
第二步,在stringA最后拼接上key得到stringSignTemp字符串,并对stringSignTemp进行MD5运算,再将得到的字符串所有字符转换为大写,得到sign值signValue。
key设置路径:微信商户平台(pay.weixin.qq.com)-->账户设置-->API安全-->密钥设置
举例:
假设传送的参数如下:
appid: wxd930ea5d5a258f4f
mch_id: 10000100
device_info: 1000
body: test
nonce_str: ibuaiVcKdpRxkhJA
第一步:对参数按照key=value的格式,并按照参数名ASCII字典序排序如下:
stringA="appid=wxd930ea5d5a258f4f&body=test&device_info=1000&mch_id=10000100&nonce_str=ibuaiVcKdpRxkhJA";
第二步:拼接API密钥:
stringSignTemp="stringA&key=192006250b4c09247ec02edce69f6a2d"
sign=MD5(stringSignTemp).toUpperCase()="9A0A8659F005D6984697E2CA0A9CF3B7"
最终得到最终发送的数据:
<xml>
<appid>wxd930ea5d5a258f4f</appid>
<mch_id>10000100</mch_id>
<device_info>1000<device_info>
<body>test</body>
<nonce_str>ibuaiVcKdpRxkhJA</nonce_str>
<sign>9A0A8659F005D6984697E2CA0A9CF3B7</sign>
<xml>
微信提供相关接口在线签名验证工具:点击进入。
所以,如果我们增加两个参数:attach、device_info就必须根据参数名ASCII字典序增加,如下图位置。如果将参数attach放在trade_type之后肯定就会报签名错误。
List<NameValuePair> packageParams = new LinkedList<NameValuePair>();
packageParams.add(new BasicNameValuePair("appid", Constant.weixin_appID));
packageParams.add(new BasicNameValuePair("attach", mOrderNo));
packageParams.add(new BasicNameValuePair("body", "订单-" + mOrderNo)); //商品描述
packageParams.add(new BasicNameValuePair("device_info", "ANDROID"));
packageParams.add(new BasicNameValuePair("mch_id", Constant.MCH_ID));
packageParams.add(new BasicNameValuePair("nonce_str", nonceStr));
packageParams.add(new BasicNameValuePair("notify_url", "http://wap.baidu.com/wechat_notify.php"));
packageParams.add(new BasicNameValuePair("out_trade_no",genOutTradNo()));
packageParams.add(new BasicNameValuePair("spbill_create_ip", "127.0.0.1"));
packageParams.add(new BasicNameValuePair("total_fee", String.valueOf((int) (mMoney * 100))));//商品金额,以分为单位
packageParams.add(new BasicNameValuePair("trade_type", "APP"));
最后再为大家分享点干货。
在计算订单总价的时候,有时会出现***.09999999999999998的现象,这个是浮点数计算出现的问题,得需要 BigDecimal解决高精度计算。
下面是我的解决方法,大家有更好的可以分享下。
-
-
-
-
- public double calculatingTotalPriceNoFreight() {
- double total = 0.00;
- for (ShopCartItem cartItem : getDataList()) {
- double orderPrice = calculatingOrderPrice(cartItem);
- total += orderPrice;
- }
-
-
- return MathUtil.round(total, 2, BigDecimal.ROUND_HALF_EVEN) ;
- }
MathUtil.Java
- package com.haier.cabinet.customer.util;
-
- import java.math.BigDecimal;
-
- public class MathUtil {
-
-
-
-
-
-
-
-
- public static double round(double value, int scale,
- int roundingMode) {
- BigDecimal bd = new BigDecimal(value);
- bd = bd.setScale(scale, roundingMode);
- double d = bd.doubleValue();
- bd = null;
- return d;
- }
-
-
-
-
-
-
-
-
- public static double sum(double d1,double d2){
- BigDecimal bd1 = new BigDecimal(Double.toString(d1));
- BigDecimal bd2 = new BigDecimal(Double.toString(d2));
- return bd1.add(bd2).doubleValue();
- }
-
-
-
-
-
-
-
-
- public static double sub(double d1,double d2){
- BigDecimal bd1 = new BigDecimal(Double.toString(d1));
- BigDecimal bd2 = new BigDecimal(Double.toString(d2));
- return bd1.subtract(bd2).doubleValue();
- }
-
-
-
-
-
-
-
- public static double mul(double d1,double d2){
- BigDecimal bd1 = new BigDecimal(Double.toString(d1));
- BigDecimal bd2 = new BigDecimal(Double.toString(d2));
- return bd1.multiply(bd2).doubleValue();
- }
-
-
-
-
-
-
-
- public static double mul(int n,double d2){
- BigDecimal bd1 = new BigDecimal(Integer.toString(n));
- BigDecimal bd2 = new BigDecimal(Double.toString(d2));
- return bd1.multiply(bd2).doubleValue();
- }
-
-
-
-
-
-
-
-
-
- public static double div(double d1,double d2,int scale){
-
-
-
- BigDecimal bd1 = new BigDecimal(Double.toString(d1));
- BigDecimal bd2 = new BigDecimal(Double.toString(d2));
- return bd1.divide
- (bd2, scale, BigDecimal.ROUND_HALF_UP).doubleValue();
- }
-
-
-
-
-
-
- public static String double2String(double d){
- BigDecimal bg = new BigDecimal(d * 100);
- double doubleValue = bg.setScale(2,BigDecimal.ROUND_HALF_UP).doubleValue();
- return String.valueOf((int)doubleValue);
- }
-
- }
2016-08-30补充:
微信官网上对于支付返回-1是这样的描述的, 可能的原因:签名错误、未注册APPID、项目设置APPID不正确、注册的APPID与设置的不匹配、其他异常等。
我犯的错和他描述的都不一样。
1、新建一个微信支付demo的工程,将包名和签名上传给微信。并保证apk是通过keystore方式打了签名的,用官方的工具做签名。
2、将官方的demo示例参考一下,
权限部分:
-
- <uses-permission android:name="android.permission.INTERNET" />
- <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS"/>
- <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
- <activity android:name=".MainActivity"
- android:exported="true">
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- <intent-filter>
- <action android:name="android.intent.action.VIEW"/>
- <category android:name="android.intent.category.DEFAULT"/>
- <data android:scheme="wxd31ef1f4dd******"/>
- </intent-filter>
- </activity>
- <activity android:name=".AlipayH5Activity"></activity>
- <activity android:name="com.*******.paydemo.wxapi.WXPayEntryActivity"
- android:exported="true"
- android:launchMode="singleTop">
- </activity></span>
Activity部分的逻辑代码:
- IWXAPI api = WXAPIFactory.createWXAPI(MainActivity.this, null);
- api.registerApp(ConstUtil.APP_ID);
- PayReq request = new PayReq();
- request.appId = ConstUtil.APP_ID;
- request.partnerId = ConstUtil.PARTNER_ID;
- request.prepayId = wxPrepayEntity.getPrepay_id();
- request.nonceStr = wxPrepayEntity.getReq_noncestr();
- request.timeStamp = wxPrepayEntity.getReq_timestamp();
- request.packageValue = wxPrepayEntity.getReq_package();
- request.sign = wxPrepayEntity.getReq_sign();
-
-
- api.sendReq(request);
每一个参数都很关键的,如果签名和包名确定都没有问题,那么基本上是传递的参数有问题。最好跟做后台的同事一起看看就能解决问题。
比如partnerId传错了就会出现签名错误,再比如timeStamp,你用了自定义的时间戳与服务器的不一致,也会出现错误。
排查错误最好把sign这个字段的值打印出来,与服务器生成的信息做比较,参数值没有问题才会不出问题!