引言
集成微信SDK就不废话了,在微信第三方登录做了介绍,传送门:微信第三方登录
签名
微信支付个人觉得最难的就是签名这个概念。其实就是拼接参数然后MD5加密得到的一串字符串一样的东西。而当我们要去请求微信官方的接口必须带上签名这个参数,关于签名的校验官方也提供了校验工具微信官方校验工具
1.统一下单
- URL地址:https://api.mch.weixin.qq.com/pay/unifiedorder
- 大概意思就是要传这些参数给微信,统一下单所需的参数中的 sign 我们叫一次签名。
1.请求统一下单接口如果成功,会返回预支付订单号prepay_id
2.调起支付接口
- 在开发中,我们有签名的地方一般都会放在后台去处理,所以下面代码是我请求我们服务器,让后台去统一下单,返回给我调起微信支付接口所需要的参数。
func getData(_ oids: String){
let url = "http://m.tdamm.com/goods/pay/paydo?oids=\(oids)&pay_type=wxpay"
RequestTool.POSTRequestWith(url, parameters: nil, success: { (task, data) in
let dict = data as! NSDictionary
var code = dict.object(forKey: "code") as! Int
if code == 200{
let pay_info = (dict.object(forKey: "data") as! NSDictionary).object(forKey: "pay_info") as! NSDictionary
let prepay_id = pay_info.object(forKey: "prepay_id") as! String
let sign = (dict.object(forKey: "data") as! NSDictionary).object(forKey: "sign") as! String
let noncestr = (dict.object(forKey: "data") as! NSDictionary).object(forKey: "noncestr") as! String
let time = (dict.object(forKey: "data") as! NSDictionary).object(forKey: "timestamp") as! Int
//调起微信
let req = PayReq()
//应用的AppID(固定的)
req.openID = WXAppID
//商户号(固定的)
req.partnerId = WXPartnerID
//扩展字段(固定的)
req.package = "Sign=WXPay"
//统一下单返回的预支付交易会话ID
req.prepayId = prepay_id
//随机字符串
req.nonceStr = noncestr
//时间戳(10位)
req.timeStamp = UInt32(time)
//签名
req.sign = sign
WXApi.send(req)
}
}) { (task, error) in
print(error)
}
}
- 从代码来看,很清楚IOS调起微信支付接口需要7个参数,分别是
//应用的AppID(固定的)
req.openID = WXAppID
//商户号(固定的)
req.partnerId = WXPartnerID
//扩展字段(固定的)
req.package = "Sign=WXPay"
//统一下单返回的预支付交易会话ID
req.prepayId = prepay_id
//随机字符串
req.nonceStr = noncestr
//时间戳(10位)
req.timeStamp = UInt32(time)
//签名
req.sign = sign
而这次要用到的sign我们称为二次签名,因为一开始后台不知道生成这个sign要用什么参数,所以我自己在iOS端自己MD5加密了。下面是我网上找的MD5加密扩展方法。
extension String {
// MD5 加密字符串
var MD5: String {
let cStr = self.cString(using: .utf8);
let buffer = UnsafeMutablePointer<UInt8>.allocate(capacity: 16)
CC_MD5(cStr!,(CC_LONG)(strlen(cStr!)), buffer)
let md5String = NSMutableString()
for i in 0..<16 {
md5String.appendFormat("%02x", buffer[i])
}
free(buffer)
return md5String as String
}
}
let strA = "appid=\(WXAppID)&noncestr=\(noncestr)&package=Sign=WXPay&partnerid=\(WXPartnerID)&prepayid=\(prepay_id)×tamp=\(time)"
req.sign = ("\(strA)&key=\(key)").MD5.uppercased()
- 通过上面的加密代码,第二次签名生成所需的参数就一目了然了,注意顺序和大小写,一共是7个参数,分别是appid,noncestr,package,partnerid,prepayid,timestamp,key(其中的key是你的商户秘钥,官方给出的key设置路径:微信商户平台(pay.weixin.qq.com)-->账户设置-->API安全-->密钥设置)。不得不吐槽的就是,当初给我的key是错误的,导致一直弹签名失败的提示框,真的很坑。
微信官方校验工具通过这个工具,如果和你生成的sign一样,那你就要看下是不是有的参数错了。
3.最后在你的AppDeleate中加入微信的回调方法
- 当然也别忘记了注册微信(注册了就无视)
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
//MARK: -注册微信
WXApi.registerApp(WXAppID)
return true
}
// 微信回调
func onResp(_ resp: BaseResp!) {
var strTitle = "支付结果"
var strMsg = "what:\(resp.errCode)"
print(resp.errCode)
// 微信支付回调
if resp.isKind(of: PayResp.self)
{
print("retcode = \(resp.errCode), retstr = \(resp.errStr)")
switch resp.errCode
{
// 支付成功
case 0 :
NotificationCenter.default.post(name: NSNotification.Name(rawValue: "WXPaySuccessNotification"), object: nil)
// 支付失败
default:
WXPayFail()
}
}
// 微信登录回调
if resp.errCode == 0 && resp.type == 0{//授权成功
let response = resp as! SendAuthResp
NotificationCenter.default.post(name: NSNotification.Name(rawValue: "WXLoginSuccessNotification"), object: response.code)
}
}
总结
总体来说代码很简单,如果后台把调用统一下单接口,二次签名之后,直接返回给你对应的参数,你直接拿后台返回给你的参数,一一对应的微信调起接口的参数就可以了。但是由于后台一开始没做过,所以我也当系统的了解一下微信支付的流程吧。