android微信支付问题总结


http://blog.csdn.net/jdsjlzx/article/details/47422279

http://blog.csdn.net/jdsjlzx/article/details/47422279

http://blog.csdn.net/jdsjlzx/article/details/47422279

 

android微信支付问题总结

  40049人阅读  评论(35)  收藏  举报
  分类:

首先我们先看一个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哈哈~)

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. @Override  
  2.    public void onResp(BaseResp resp) {  
  3.        Log.d(TAG, "onPayFinish, errCode = " + resp.errCode);  
  4.   
  5.        if (resp.getType() == ConstantsAPI.COMMAND_PAY_BY_WX) {  
  6.            AlertDialog.Builder builder = new AlertDialog.Builder(this);  
  7.            builder.setTitle(R.string.app_tip);  
  8.            builder.setMessage(getString(R.string.pay_result_callback_msg, resp.errStr +";code=" + String.valueOf(resp.errCode)));  
  9.            builder.show();  
  10.        }  
  11.    }  

3.3 生成prepay_id,在服务器完成,由服务器去跟微信服务器交互,客户端不需要参与

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. /** 
  2.    * 获取预支付订单号: 
  3.    * prepay_id(服务器完成)!!! 
  4.    * 注意:如果服务端开发文档跟客户端demo里的参数不一样,以demo里的参数为准, 
  5.    * 否则服务器传过来的参数无法调起微信支付!!! 
  6.    * */  
  7.    private String genProductArgs() {  
  8.        StringBuffer xml = new StringBuffer();  
  9.   
  10.        try {  
  11.            String  nonceStr = genNonceStr();  
  12.   
  13.   
  14.            xml.append("</xml>");  
  15.           List<NameValuePair> packageParams = new LinkedList<NameValuePair>();  
  16.            packageParams.add(new BasicNameValuePair("appid", Constants.APP_ID));  
  17.            packageParams.add(new BasicNameValuePair("body""APP pay test"));  
  18.            /**这里用的是mach_id,跟sign签名时参数名不同,一定要注意*/  
  19.            packageParams.add(new BasicNameValuePair("mch_id", Constants.MCH_ID));  
  20.            packageParams.add(new BasicNameValuePair("nonce_str", nonceStr));  
  21.            packageParams.add(new BasicNameValuePair("notify_url""http://121.40.35.3/test"));  
  22.            packageParams.add(new BasicNameValuePair("out_trade_no",genOutTradNo()));  
  23.            packageParams.add(new BasicNameValuePair("spbill_create_ip","127.0.0.1"));  
  24.            packageParams.add(new BasicNameValuePair("total_fee""1"));  
  25.            packageParams.add(new BasicNameValuePair("trade_type""APP"));  
  26.   
  27.   
  28.            String sign = genPackageSign(packageParams);  
  29.            packageParams.add(new BasicNameValuePair("sign", sign));  
  30.   
  31.   
  32.           String xmlstring =toXml(packageParams);  
  33.   
  34.            return xmlstring;  
  35.   
  36.        } catch (Exception e) {  
  37.            Log.e(TAG, "genProductArgs fail, ex = " + e.getMessage());  
  38.            return null;  
  39.        }  
  40.   
  41.   
  42.    }  

3.4 获取二次签名sign

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. private void genPayReq() {  
  2.   
  3.         req.appId = Constants.APP_ID;  
  4.         req.partnerId = Constants.MCH_ID;  
  5.         req.prepayId = resultunifiedorder.get("prepay_id");  
  6. //      req.packageValue = "prepay_id="+resultunifiedorder.get("prepay_id");  
  7.         req.packageValue = "Sign=WXPay";  
  8.         req.nonceStr = genNonceStr();  
  9.         req.timeStamp = String.valueOf(genTimeStamp());  
  10.   
  11.   
  12.         List<NameValuePair> signParams = new LinkedList<NameValuePair>();  
  13.         signParams.add(new BasicNameValuePair("appid", req.appId));  
  14.         signParams.add(new BasicNameValuePair("noncestr", req.nonceStr));  
  15.         /** 
  16.          * 这里的package参数值必须是Sign=WXPay,否则IOS端调不起微信支付, 
  17.          * (参数值是"prepay_id="+resultunifiedorder.get("prepay_id")的时候Android可以,IOS不可以) 
  18.          */  
  19.         signParams.add(new BasicNameValuePair("package", req.packageValue));  
  20.         /**注意二次签名这里不再是mch_id,变成了prepayid;*/  
  21.         signParams.add(new BasicNameValuePair("partnerid", req.partnerId));  
  22.         signParams.add(new BasicNameValuePair("prepayid", req.prepayId));  
  23.         signParams.add(new BasicNameValuePair("timestamp", req.timeStamp));  
  24.   
  25.         req.sign = genAppSign(signParams);  
  26.   
  27.         sb.append("sign\n"+req.sign+"\n\n");  
  28.   
  29.         show.setText(sb.toString());  
  30.   
  31.         Log.e("orion", signParams.toString());  
  32.   
  33.     }  

3.5 起调微信支付

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. private void sendPayReq() {  
  2.         msgApi.registerApp(Constants.APP_ID);  
  3.         msgApi.sendReq(req);  
  4.     }  

3.6 配置Manifest.xml,权限什么的按照文档的配置就行了

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <activity  
  2.            android:name=".PayActivity"  
  3.            android:label="@string/app_name"  
  4.            android:exported="true"  
  5.            android:launchMode="singleTop">  
  6.            <intent-filter>  
  7.                <action android:name="android.intent.action.MAIN" />  
  8.                <category android:name="android.intent.category.LAUNCHER" />  
  9.            </intent-filter>  
  10.            <!--这个intent-filter不要忘了-->  
  11.            <intent-filter>  
  12.                <action android:name="android.intent.action.VIEW"/>  
  13.                <category android:name="android.intent.category.DEFAULT"/>  
  14.                <data android:scheme="wxd930ea5d5a258f4f"/>  
  15.            </intent-filter>  
  16.        </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。

特别注意以下重要规则:

  1. ◆ 参数名ASCII码从小到大排序(字典序);
  2. ◆ 如果参数的值为空不参与签名;
  3. ◆ 参数名区分大小写;
  4. ◆ 验证调用返回或微信主动通知签名时,传送的sign参数不参与签名,将生成的签名与该sign值作校验。
  5. ◆ 微信接口可能增加字段,验证签名时必须支持增加的扩展字段

第二步,在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解决高精度计算。

下面是我的解决方法,大家有更好的可以分享下。

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. /* 
  2.      * 总订单的价格(不包含运费) 
  3.      * @return 
  4.      */  
  5.     public double calculatingTotalPriceNoFreight() {  
  6.         double total = 0.00;  
  7.         for (ShopCartItem cartItem : getDataList()) {  
  8.             double orderPrice = calculatingOrderPrice(cartItem);  
  9.             total += orderPrice;  
  10.         }  
  11.   
  12.         //double值保留 2 位小数,使用银行家舍入法  
  13.         return MathUtil.round(total, 2, BigDecimal.ROUND_HALF_EVEN) ;  
  14.     }  


MathUtil.Java

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. package com.haier.cabinet.customer.util;  
  2.   
  3. import java.math.BigDecimal;  
  4.   
  5. public class MathUtil {  
  6.   
  7.     /**    
  8.      * 对double数据进行取精度.    
  9.      * @param value  double数据.    
  10.      * @param scale  精度位数(保留的小数位数).    
  11.      * @param roundingMode  精度取值方式.    
  12.      * @return 精度计算后的数据.    
  13.      */     
  14.     public static double round(double value, int scale,    
  15.              int roundingMode) {      
  16.         BigDecimal bd = new BigDecimal(value);      
  17.         bd = bd.setScale(scale, roundingMode);    
  18.         double d = bd.doubleValue();      
  19.         bd = null;      
  20.         return d;      
  21.     }      
  22.   
  23.   
  24.      /**  
  25.      * double 相加  
  26.      * @param d1  
  27.      * @param d2  
  28.      * @return  
  29.      */   
  30.     public static double sum(double d1,double d2){   
  31.         BigDecimal bd1 = new BigDecimal(Double.toString(d1));   
  32.         BigDecimal bd2 = new BigDecimal(Double.toString(d2));   
  33.         return bd1.add(bd2).doubleValue();   
  34.     }   
  35.   
  36.   
  37.     /**  
  38.      * double 相减  
  39.      * @param d1  
  40.      * @param d2  
  41.      * @return  
  42.      */   
  43.     public static double sub(double d1,double d2){   
  44.         BigDecimal bd1 = new BigDecimal(Double.toString(d1));   
  45.         BigDecimal bd2 = new BigDecimal(Double.toString(d2));   
  46.         return bd1.subtract(bd2).doubleValue();   
  47.     }   
  48.   
  49.     /**  
  50.      * double 乘法  
  51.      * @param d1  
  52.      * @param d2  
  53.      * @return  
  54.      */   
  55.     public static double mul(double d1,double d2){   
  56.         BigDecimal bd1 = new BigDecimal(Double.toString(d1));   
  57.         BigDecimal bd2 = new BigDecimal(Double.toString(d2));   
  58.         return bd1.multiply(bd2).doubleValue();   
  59.     }   
  60.       
  61.     /**  
  62.      * double 乘法  
  63.      * @param n 
  64.      * @param d2  
  65.      * @return  
  66.      */   
  67.     public static double mul(int n,double d2){   
  68.         BigDecimal bd1 = new BigDecimal(Integer.toString(n));   
  69.         BigDecimal bd2 = new BigDecimal(Double.toString(d2));   
  70.         return bd1.multiply(bd2).doubleValue();   
  71.     }   
  72.   
  73.   
  74.     /**  
  75.      * double 除法  
  76.      * @param d1  
  77.      * @param d2  
  78.      * @param scale 四舍五入 小数点位数  
  79.      * @return  
  80.      */   
  81.     public static double div(double d1,double d2,int scale){   
  82.         //  当然在此之前,你要判断分母是否为0,      
  83.         //  为0你可以根据实际需求做相应的处理   
  84.   
  85.         BigDecimal bd1 = new BigDecimal(Double.toString(d1));   
  86.         BigDecimal bd2 = new BigDecimal(Double.toString(d2));   
  87.         return bd1.divide  
  88.                 (bd2, scale, BigDecimal.ROUND_HALF_UP).doubleValue();  
  89.     }  
  90.   
  91.     /** 
  92.      * 将double类型数据转为字符串 
  93.      * @param d 
  94.      * @return 
  95.      */  
  96.     public static String double2String(double d){  
  97.         BigDecimal bg = new BigDecimal(d * 100);  
  98.         double doubleValue = bg.setScale(2,BigDecimal.ROUND_HALF_UP).doubleValue();  
  99.         return  String.valueOf((int)doubleValue);  
  100.     }  
  101.   
  102. }  


2016-08-30补充:

微信官网上对于支付返回-1是这样的描述的,  可能的原因:签名错误、未注册APPID、项目设置APPID不正确、注册的APPID与设置的不匹配、其他异常等。

我犯的错和他描述的都不一样。

1、新建一个微信支付demo的工程,将包名和签名上传给微信。并保证apk是通过keystore方式打了签名的,用官方的工具做签名。

2、将官方的demo示例参考一下,

权限部分:

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <!-- WeixinPay -->  
  2.     <uses-permission android:name="android.permission.INTERNET" />  
  3.     <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS"/>  
  4.     <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"  

[html]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <activity android:name=".MainActivity"  
  2.      android:exported="true">  
  3.      <intent-filter>  
  4.          <action android:name="android.intent.action.MAIN" />  
  5.          <category android:name="android.intent.category.LAUNCHER" />  
  6.      </intent-filter>  
  7.      <intent-filter>  
  8.          <action android:name="android.intent.action.VIEW"/>  
  9.          <category android:name="android.intent.category.DEFAULT"/>  
  10.          <data android:scheme="wxd31ef1f4dd******"/>  
  11.      </intent-filter>  
  12.  </activity>  
  13.  <activity android:name=".AlipayH5Activity"></activity>  
  14.  <activity android:name="com.*******.paydemo.wxapi.WXPayEntryActivity"  
  15.      android:exported="true"  
  16.      android:launchMode="singleTop">  
  17.  </activity></span>  

Activity部分的逻辑代码:

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. IWXAPI api = WXAPIFactory.createWXAPI(MainActivity.thisnull);//通过工厂创建对象  
  2.                                 api.registerApp(ConstUtil.APP_ID);  
  3.                                 PayReq request = new PayReq();  
  4.                                 request.appId = ConstUtil.APP_ID;  
  5.                                 request.partnerId = ConstUtil.PARTNER_ID;  
  6.                                 request.prepayId = wxPrepayEntity.getPrepay_id();  
  7.                                 request.nonceStr = wxPrepayEntity.getReq_noncestr();  
  8.                                 request.timeStamp = wxPrepayEntity.getReq_timestamp();  
  9.                                 request.packageValue = wxPrepayEntity.getReq_package();  
  10.                                 request.sign = wxPrepayEntity.getReq_sign();  
  11. //                                request.extData = "app data"; // optional  
  12.                                 // 在支付之前,如果应用没有注册到微信,应该先调用IWXMsg.registerApp将应用注册到微信  
  13.                                 api.sendReq(request);  
每一个参数都很关键的,如果签名和包名确定都没有问题,那么基本上是传递的参数有问题。最好跟做后台的同事一起看看就能解决问题。

比如partnerId传错了就会出现签名错误,再比如timeStamp,你用了自定义的时间戳与服务器的不一致,也会出现错误。

排查错误最好把sign这个字段的值打印出来,与服务器生成的信息做比较,参数值没有问题才会不出问题!


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值