您的位置:首页 > 移动开发 > 微信开发

android微信支付(问题总结)

2015-11-30 17:10 621 查看
感谢博主的分享:http://blog.csdn.net/baidu_17508977/article/details/44517283

onPayFinish, errCode = -1

当你参数签名都没有问题的时候,出现这个提示,请按照如下操作:

在你的项目测试微信的组件(分享、支付等)的时候,一定要用你自己的keystore签名出来测试,如果用debug.keystore肯定是不成功的,浪费了大家不少的时间!

谨以此文献给苦苦挣扎在微信支付给我们挖的坑中的程序猿和媛们,百度了很久很久,网络上也没有一个成功的微信支付demo,既然我做成了,本着跟大伙分享的原则,特此奉献出我开发过程中遇到的问题,这是本人亲测的成功案例,而且是v3.0的版本,不是官网的1.7版本!!!

1,项目进展到现在,市场部要求加入微信支付功能,然后去官网下载了demo,研究了一下,同时又百度了一下,腾讯留下的各种坑。差不多花了几天时间搞懂了,谁知道,最大的坑出现了,NND,微信官网提供的demo和开发文档是1.7的,我们产品部通过客服跟腾讯竟然要到了3.0版本的,你说这还能不能愉快的玩微信支付了。好了,不吐槽腾讯了,留给大家自己体会,O(∩_∩)O哈哈~。

2,在微信开放平台申请app_id,app_key我就不在这里叙述了,稍后我会把开发文档一并上传的,你也可以去微信开放平台自行查看(差不多一个周才会通过审核)

https://open.weixin.qq.com/cgi-bin/index?t=home/index&lang=zh_CN&token=44d65f9f1df0d4725b941e217a0bf769fbb51b3a

3,下面我就来说一说微信支付开发时需要注意的地方:

3.1 首先来看一下,微信支付的架构和流程图





3.2 再来看一下成功调起微信支付的界面



4下面开始讲解配置工程

4.1 这里必须要有wxapi这个包名,同时必须有WXPayEntryActivity这个类名,否则无法调起微信支付,(开发文档没有标注,废了好大周章)



4.2 支付成功通知:在WXPayEntryActivity的OnResp中处理,不能以微信返回的通知界面为准(我遇到的情况,网络不稳定的时候,微信返回界面提示支付失败,但是收到微信通知其实已经支付成功了),必须要去自己的服务器查询支付状态,这里微信建议用轮循机制去查询(最好听微信劝,O(∩_∩)O哈哈~)

[java] view
plaincopy

@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();

}

}

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

[java] view
plaincopy

/**

* 获取预支付订单号:

* prepay_id(服务器完成)!!!

* 注意:如果服务端开发文档跟客户端demo里的参数不一样,以demo里的参数为准,

* 否则服务器传过来的参数无法调起微信支付!!!

* */

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"));

/**这里用的是mach_id,跟sign签名时参数名不同,一定要注意*/

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;

}

}

4.4 获取二次签名sign

[java] view
plaincopy

private void genPayReq() {

req.appId = Constants.APP_ID;

req.partnerId = Constants.MCH_ID;

req.prepayId = resultunifiedorder.get("prepay_id");

// req.packageValue = "prepay_id="+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));

/**

* 这里的package参数值必须是Sign=WXPay,否则IOS端调不起微信支付,

* (参数值是"prepay_id="+resultunifiedorder.get("prepay_id")的时候Android可以,IOS不可以)

*/

signParams.add(new BasicNameValuePair("package", req.packageValue));

/**注意二次签名这里不再是mch_id,变成了prepayid;*/

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());

}

4.5 起调微信支付

[java] view
plaincopy

private void sendPayReq() {

msgApi.registerApp(Constants.APP_ID);

msgApi.sendReq(req);

}

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

[html] view
plaincopy

<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不要忘了-->

<intent-filter>

<action android:name="android.intent.action.VIEW"/>

<category android:name="android.intent.category.DEFAULT"/>

<data android:scheme="wxd930ea5d5a258f4f"/>

</intent-filter>

</activity>

5,支付通知接口和退款接口按照开发文档即可,这里不再赘述

6, 在你的项目测试微信支付的时候,一定要用你自己的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.rrslj.com/wechat/wechat_pay/wechat_notify.php"));
packageParams.add(new BasicNameValuePair("out_trade_no",genOutTradNo()));
packageParams.add(new BasicNameValuePair("spbill_create_ip", "127.0.0.1"));
Log.d(TAG, "total_fee = " + String.valueOf((int) (mMoney * 100)));
packageParams.add(new BasicNameValuePair("total_fee", String.valueOf((int) (mMoney * 100))));//商品金额,以分为单位
packageParams.add(new BasicNameValuePair("trade_type", "APP"));
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: