之前做过微信支付,也遇到了一些坑,只是当时没有总结。再次使用时还要看开发文档,重新踩坑,所以有必要总结一下,为了下次快速集成。
微信官方集成文档
支付开发者文档
1.Android studio微信支付sdk可以通过gradle来导入
dependencies {
compile 'com.tencent.mm.opensdk:wechat-sdk-android-without-mta:+'
}
2.接收微信的请求及返回值(坑点)
集成文档是这样描述的
Paste_Image.png看图是并列关系,但是并非如此,应该如此
Paste_Image.png3..调起微信APP进行支付
调起支付需要传递参数,需要请求自己服务器返回如下参数:
Paste_Image.png
返回参数后,调起微信
//微信支付api
private IWXAPI iwxapi;
/**
*调起微信支付的方法
**/
private void toWXPay() {
//初始化微信api
iwxapi = WXAPIFactory.createWXAPI(this, null);
//注册appid appid可以在开发平台获取
iwxapi.registerApp(appid);
Runnable payRunnable = new Runnable() { //这里注意要放在子线程
@Override
public void run() {
PayReq request = new PayReq(); //调起微信APP的对象
//下面是设置必要的参数,也就是前面说的参数,这几个参数从何而来请看上面说明
request.appId = appid;
request.partnerId = partnerId;
request.prepayId = prepayId;
request.packageValue = "Sign=WXPay";
request.nonceStr = nonceStr;
request.timeStamp = timeStamp;
request.sign = sign;
iwxapi.sendReq(request);//发送调起微信的请求
}
};
Thread payThread = new Thread(payRunnable);
payThread.start();
}
签名sign 参数,又时需要我们客户端来生成,来看一下生成规则
Paste_Image.png
生成事例:
//签名
LinkedHashMap<String, String> signParams = new LinkedHashMap<>();
signParams.put("appid", request.appId);
signParams.put("noncestr", request.nonceStr);
signParams.put("package", request.packageValue);
signParams.put("partnerid", request.partnerId);
signParams.put("prepayid", request.prepayId);
signParams.put("timestamp", request.timeStamp);
request.sign = genPackageSign(signParams,key);
iwxapi.sendReq(request);//发送调起微信的请求
/**
* 调起微信APP支付,签名
* 生成签名
*/
private String genPackageSign(LinkedHashMap<String, String> params,String key) {
StringBuilder sb = new StringBuilder();
for (Map.Entry<String,String> entry: params.entrySet()) {
sb.append(entry.getKey());
sb.append('=');
sb.append(entry.getValue());
sb.append('&');
}
sb.append("key=");
sb.append(key);
String packageSign = getMessageDigest(sb.toString().getBytes()).toUpperCase();
return packageSign;
}
/**
* md5加密
* @param buffer
* @return
*/
private String getMessageDigest(byte[] buffer) {
char hexDigits[] = { 'a', 'b', 'c', 'd', 'e', 'f' , 'g', 'h', 'l', 'm', 'n', 'o' };
try {
MessageDigest mdTemp = MessageDigest.getInstance("MD5");
mdTemp.update(buffer);
byte[] md = mdTemp.digest();
int j = md.length;
char str[] = new char[j * 2];
int k = 0;
for (int i = 0; i < j; i++) {
byte byte0 = md[i];
str[k++] = hexDigits[byte0 >>> 4 & 0xf];
str[k++] = hexDigits[byte0 & 0xf];
}
return new String(str);
} catch (Exception e) {
return null;
}
}
/**
* 获取随机数
*
* @return
*/
private String genNonceStr() {
Random random = new Random();
return getMessageDigest(String.valueOf(random.nextInt(10000)).getBytes());
}
/**
* 获取时间戳
*
* @return
*/
private long genTimeStamp() {
return System.currentTimeMillis() / 1000;
}
4.微信支付的回调
处理微信支付的回调就是在创建的WXEntryActivity中处理。需要实现IWXAPIEventHandler接口,这个接口会要求你实现onResp方法,就在这个方法中处理回调。
@Override
public void onResp(BaseResp resp) {
if(resp.getType()==ConstantsAPI.COMMAND_PAY_BY_WX){
if(resp.errCode==0){
Toast.makeText(this, "支付成功", Toast.LENGTH_LONG).show();
}
else {
Toast.makeText(this, "支付失败", Toast.LENGTH_LONG).show();
}
finish();
}
}
网友评论