微信支付现在使用越来越广泛,最近公司的app和微信公众号也需要实现这一功能。下面主要讲解一下app微信支付的开发过程,以及容易遇到的问题。
一、商户通过微信商户平台申请,获取appid和key(通过邮件形式生成),填入包名和签名,可以通过工具直接生成签名填入,工具地址“https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5”,这一过程大概会持续一星期。
二、服务端通过appid,key等参数请求“https://api.mch.weixin.qq.com/pay/unifiedorder”从而获取参数,返回给客户端。具体参照“https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1#”。之后将参数返回给客户端。
三、android端,主要利用服务端返回的参数进行请求,主要的参数有(appId(可以写死),prepayId,partnerId ,nonceStr,packageValue,timeStamp,signPay,其中最坑爹的就是sign签名。如果sign没有获取成功就会发生,能跳转到微信,或者能登陆,但不能弹出付款界面的问题。一方面sign通过服务端的md5加密来返回给客户端相应的值。但是如果自己不放心可以自己通过一些参数来加密得到sign。
md5加密算法:
package aywx.com.application.utils.wxpay;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
/**
* Created by 曾志强 on 2016/7/28.
*/
public class MD5util {
// 全局数组
private final static String[] strDigits = { "0", "1", "2", "3", "4", "5",
"6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };
public MD5util() {
}
// 返回形式为数字跟字符串
private static String byteToArrayString(byte bByte) {
int iRet = bByte;
// System.out.println("iRet="+iRet);
if (iRet < 0) {
iRet += 256;
}
int iD1 = iRet / 16;
int iD2 = iRet % 16;
return strDigits[iD1] + strDigits[iD2];
}
// 返回形式只为数字
private static String byteToNum(byte bByte) {
int iRet = bByte;
System.out.println("iRet1=" + iRet);
if (iRet < 0) {
iRet += 256;
}
return String.valueOf(iRet);
}
// 转换字节数组为16进制字串
private static String byteToString(byte[] bByte) {
StringBuffer sBuffer = new StringBuffer();
for (int i = 0; i < bByte.length; i++) {
sBuffer.append(byteToArrayString(bByte[i]));
}
return sBuffer.toString();
}
public static String GetMD5Code(String strObj) {
String resultString = null;
try {
resultString = new String(strObj);
MessageDigest md = MessageDigest.getInstance("MD5");
// md.digest() 该函数返回值为存放哈希值结果的byte数组
resultString = byteToString(md.digest(strObj.getBytes()));
} catch (NoSuchAlgorithmException ex) {
ex.printStackTrace();
}
return resultString;
}
}
通过参数请求服务器部分代码:
PayReq payReq = new PayReq();
payReq.appId = payParam.getAppid();
payReq.prepayId=payParam.getPrepay_id();
// payReq.partnerId="wx20160727211808f92f687b9a0381369812";
payReq.nonceStr = payParam.getNonceStr();
payReq.packageValue="Sign=WXPay";
payReq.timeStamp=payParam.getTimestamp();
payReq.sign=sign;
payReq.partnerId=payParam.getMch_id();
api.sendReq(payReq);
回调按照平常网上方法就不会出错。主要就是签名。