Swift3.0集成微信支付SDK

PS:我是小白,有不正确的请多指教

最近公司的项目需要做微信和支付宝支付功能,我的项目是用Swift3.0做的,记录一下我的集成过程以及遇到的坑。
先做的是微信支付,不得不吐槽,微信支付的官方文档有点坑,好久都没更新了,遇到了不少坑。当然也许只我技术不行。调起支付的参数出了签名(sign)外别的我们都是后台弄好给我们。其余关于在微信开放平台注册等就不说了

首先是集成SDK,根据微信支付官方文档我用的是cocoapods,没有什么问题,根据前两步项目设置APPID和在appdelegate里面注册APPID(WXApi.registerApp(weixinAppId, enableMTA: true)
然后接着在appdelegate里设置回调

 //ios9之前
    func application(_ application: UIApplication, handleOpen url: URL) -> Bool {
        if url.host == "safepay"{
//这个是支付宝用的
            aliPayCallBackMethod(url: url)
            return true
        }
        return WXApi.handleOpen(url, delegate: self)
    }
    
    //ios9之后
    func application(_ application: UIApplication, open url: URL, sourceApplication: String?, annotation: Any) -> Bool {
        if url.host == "safepay"{
            aliPayCallBackMethod(url: url)
            return true
        }
         return WXApi.handleOpen(url, delegate: self)
    }
    
    func application(_ app: UIApplication, open url: URL, options: [UIApplicationOpenURLOptionsKey : Any] = [:]) -> Bool {
        if url.host == "safepay"{
           aliPayCallBackMethod(url: url)
            return true
        }
        return WXApi.handleOpen(url, delegate: self)
    }

上面三个方法我用到时候发现之后调用第三个,但是我都写上去了。
然后设置appdelegate遵守WXApiDelegate协议
代理方法是支付结果的一个回调

 func onResp(_ resp: BaseResp!) {
        var detailMessage = String()
        if resp.isKind(of: PayResp.self){
            //支付返回的结果
            switch resp.errCode {
            case WXSuccess.rawValue:
                detailMessage = "支付结果:成功!"
                NotificationCenter.default.post(name: NSNotification.Name(rawValue: weixinPayResultNotiftcation), object: "success")
                break
            default:
                detailMessage = "支付结果:失败!" + "errCode\(resp.errCode)" + "errStr\(resp.errStr)"
                MyLog(message: detailMessage)
                NotificationCenter.default.post(name: NSNotification.Name(rawValue: weixinPayResultNotiftcation), object: "fail")
                break
            }
        }
    }

这样appdelegate里的工作就结束了,接下来就是点击支付按钮进行支付的那个控制器里进行下面的工作了。
刚才那个支付结果的回调结果发的通知在这里接收。注册通知

override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        NotificationCenter.default.addObserver(self, selector: #selector(getWeiXinPayResultNotificationMethod(noti:)), name: NSNotification.Name(rawValue: weixinPayResultNotiftcation), object: nil)
    }
    
    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        NotificationCenter.default.removeObserver(self)
    }

接着就是点击微信支付按钮的点击事件了

 @objc fileprivate func submitBtnClick(){
              //微信
                if !(WXApi.isWXAppInstalled()) || !(WXApi.isWXAppSupport()){
                MySVProgressHUD.show(message: "您未安装微信或者版本不支持", status: .error)
                    return
                }
                 weixinPayMethod()   
    }

//微信支付
    fileprivate func weixinPayMethod(){
            MySVProgressHUD.show(message: "正在为您支付!", status: .status)

//下面这个方法就是向自己的服务器请求发起支付需要的数据
            getWeiXinPayNeedDataMethod()
    }

 //请求数据,解析数据
    fileprivate func getWeiXinPayNeedDataMethod(){
        let nameArr = []
        let valueArr = []
        NetWorkTools.postWithURLString(URLString: URL, nameArray: nameArr, valueArray: valueArr) { (dict, error) in
       '''省略。。。
                weakSelf?.dealDataWithDic(dic: orderInfo)
            }else{
            MySVProgressHUD.show(message: netError, status: .error)
            }
      }
  }

 fileprivate func dealDataWithDic(dic : [String : Any]){
// ##这里我把数据处理的代码放到WeiXinPayManager里面去了,TimesOperations.getNetworkTime()是获取当前时间戳,在这里获取一次就行了,带到后面去后面会用到
    WeiXinPayManager.weiXinPayMethodWithDic(dic: dic, payTimeStr: TimesOperations.getNetworkTime())
    }

这个控制器里还有一个刚才接收到接受支付结果通知的观察者方法
 @objc fileprivate func getWeiXinPayResultNotificationMethod(noti : NSNotification){
        let resultMessage = noti.object as! String
        if resultMessage == "success"{
            self.showAlertMessage(title: "恭喜", message: "您已成功支付啦!")
        }else{
            self.showAlertMessage(title: "提示", message: "支付失败!")
        }
    }
 fileprivate func showAlertMessage(title : String,message : String){
        let alertVC = UIAlertController(title: title, message: message, preferredStyle: .alert)
        let sureAction = UIAlertAction(title: "OK", style: .default) { (action) in
           //进行一些支付完成后的操作,比如刷新数据,移除一些遮罩View等
        }
        sureAction.setValue(RGB(R: 107, G: 107, B: 199), forKey: "titleTextColor")
        alertVC.addAction(sureAction)
        self.present(alertVC, animated: true, completion: nil)
    }

接下来就是数据处理的部分了,坑就在这里,主要是生成一个签名Sign麻烦,去到WeiXinPayManager类里面
通过传过去的字典,获取到partnerId,prepayId,nonceStr,我方法比较笨拙,别见笑,是直接去取得

    class func weiXinPayMethodWithDic(dic : [String : Any],payTimeStr : String?){
        let partnerId = dic["mch_id"] as? String
        let prepayId = dic["prepay_id"] as? String
        let nonceStr = dic["nonce_str"] as? String

//下面这个sign就是我们需要的,我是下面用了一个方法返回的,package是微信官方文档要求的这么写的值为"Sign=WXPay”
        let sign = WeiXinPayManager().getWeiXinSignStrMethod(dic: dic, payTimeStr: payTimeStr)

        if partnerId != nil,prepayId != nil,nonceStr != nil,payTimeStr != nil,sign != nil{
            let time = (payTimeStr! as NSString).intValue
            WeiXinPayManager().requestWeiXinPayMethod(partnerId: partnerId!, prepayId: prepayId!, package: "Sign=WXPay", nonceStr: nonceStr!, timeStamp: UInt32(time), sign: sign!)
        }else{
            MySVProgressHUD.show(message: "参数出错啦", status: .error)
        }
    }
    
    fileprivate func requestWeiXinPayMethod(partnerId : String,prepayId : String,package : String,nonceStr : String,timeStamp : UInt32,sign : String){
        MySVProgressHUD.dismiss()
        let request = PayReq()
        request.partnerId = partnerId
        request.prepayId = prepayId
        request.package = package
        request.nonceStr = nonceStr
        request.timeStamp = timeStamp
        request.sign = sign
        WXApi.send(request)
    }

到这差不多就可以经行支付了。当然还有一个重要的Sign的生成还没写
关于生成Sign,官方文档是这么写的:

第一步设所有发送或者接收到的数据为集合M,将集合M内非空参数值的参数按照参数名ASCII码从小到大排序(字典序),使用URL键值对的格式(即key1=value1&key2=value2…)拼接成字符串stringA。

虽然下面也列举了一些参数,但是这里并没有真正告诉我们要传哪些,根据他列举的参数我搞了好久,毛用没有,都是只出现一个白色确定按钮,这个就说明是这个签名Sign错了,后来还是安卓的同事提示慢慢试出来了,注意参数是"appid","noncestr","prepayid",package,timestamp,partnerid,这6个,其中partnerid是商户id,可以让后台返回,也可以在自己申请的微信支付平台账号中找到,直接写死并且所有的参数不用加下划线,不要像官方文档那样使用mch_id这种拼接,注意mch_id用不上,下面就是要将参数按照从小到大排序(文档要求的)然后拼接了,(sign也可以服务端生成,服务端调用统一下单接口后获取到预支付id,再进行二次签名得到我们需要的sign:参考(https://blog.csdn.net/sinat_17820239/article/details/51334181?utm_source=blogxgwz9))因为我是直接从后台给的数据里去取参数,需要过滤不需要的key,也许把问题弄复杂了,反正这里要做的就是将这些Key从小到大排序,对应的Value也是一样,让后按照”key1=value1&key2=value2…“拼接。从小到大排序我用的是swift中的方法
var sortedKeyArray = newKeys.sorted { ( key1 : String, key2 : String) -> Bool in return key1 < key2 }
每个人拼接的方法思路不一样,我就不献丑了,好了,假设把这6个Key和对应的Value按照URL键值对的格式(即key1=value1&key2=value2…)拼接成了stringA,接着拼接行一个申请得到的weixinKey,这个key在微信支付开发者账号里找

 fileprivate func getWeiXinSignStrMethod(dic : [String : Any],payTimeStr : String?)->String?{

//1.先自己拼接得到了stringA,假设是stringA = "appid="123"&noncestr="123&package="Sign=WXPay"&partnerid="123"&prepayid="123"&timestamp= payTimeStr!"

//2.接着拼接weixinKey
stringA.append("&key=")  
stringA.append(weixinKey)

//3.把新得到的stringA进行MD5加密,再把得到的MD5值中所有的字母变成大写的,使用NSString的方法uppercased
if MD5Hash.get32MD5Hash(stringA as String!) != nil{
            return (MD5Hash.get32MD5Hash(contentString as String!)! as NSString).uppercased
        }else{
            return nil
        }
}

至此,Swift微信支付就做完了,技术有限,有做的不对的请指教!下篇记录关于Swift支付宝支付的集成

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容