Swift3.0集成微信支付SDK

PS:我是小白耘擂,有不正確的請多指教

最近公司的項目需要做微信和支付寶支付功能蓖租,我的項目是用Swift3.0做的,記錄一下我的集成過程以及遇到的坑饭玲。
先做的是微信支付侥祭,不得不吐槽,微信支付的官方文檔有點坑茄厘,好久都沒更新了矮冬,遇到了不少坑。當(dāng)然也許只我技術(shù)不行次哈。調(diào)起支付的參數(shù)出了簽名(sign)外別的我們都是后臺弄好給我們胎署。其余關(guān)于在微信開放平臺注冊等就不說了

首先是集成SDK,根據(jù)微信支付官方文檔我用的是cocoapods,沒有什么問題,根據(jù)前兩步項目設(shè)置APPID和在appdelegate里面注冊APPID(WXApi.registerApp(weixinAppId, enableMTA: true)
然后接著在appdelegate里設(shè)置回調(diào)

 //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)
    }

上面三個方法我用到時候發(fā)現(xiàn)之后調(diào)用第三個窑滞,但是我都寫上去了琼牧。
然后設(shè)置appdelegate遵守WXApiDelegate協(xié)議
代理方法是支付結(jié)果的一個回調(diào)

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

這樣appdelegate里的工作就結(jié)束了,接下來就是點擊支付按鈕進行支付的那個控制器里進行下面的工作了哀卫。
剛才那個支付結(jié)果的回調(diào)結(jié)果發(fā)的通知在這里接收巨坊。注冊通知

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)

//下面這個方法就是向自己的服務(wù)器請求發(fā)起支付需要的數(shù)據(jù)
            getWeiXinPayNeedDataMethod()
    }

 //請求數(shù)據(jù),解析數(shù)據(jù)
    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]){
// ##這里我把數(shù)據(jù)處理的代碼放到WeiXinPayManager里面去了,TimesOperations.getNetworkTime()是獲取當(dāng)前時間戳共啃,在這里獲取一次就行了占调,帶到后面去后面會用到
    WeiXinPayManager.weiXinPayMethodWithDic(dic: dic, payTimeStr: TimesOperations.getNetworkTime())
    }

這個控制器里還有一個剛才接收到接受支付結(jié)果通知的觀察者方法
 @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
           //進行一些支付完成后的操作暂题,比如刷新數(shù)據(jù),移除一些遮罩View等
        }
        sureAction.setValue(RGB(R: 107, G: 107, B: 199), forKey: "titleTextColor")
        alertVC.addAction(sureAction)
        self.present(alertVC, animated: true, completion: nil)
    }

接下來就是數(shù)據(jù)處理的部分了究珊,坑就在這里薪者,主要是生成一個簽名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: "參數(shù)出錯啦", 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)
    }

到這差不多就可以經(jīng)行支付了。當(dāng)然還有一個重要的Sign的生成還沒寫
關(guān)于生成Sign,官方文檔是這么寫的:

第一步設(shè)所有發(fā)送或者接收到的數(shù)據(jù)為集合M瑟蜈,將集合M內(nèi)非空參數(shù)值的參數(shù)按照參數(shù)名ASCII碼從小到大排序(字典序)烟逊,使用URL鍵值對的格式(即key1=value1&key2=value2…)拼接成字符串stringA。

雖然下面也列舉了一些參數(shù)铺根,但是這里并沒有真正告訴我們要傳哪些宪躯,根據(jù)他列舉的參數(shù)我搞了好久,毛用沒有位迂,都是只出現(xiàn)一個白色確定按鈕访雪,這個就說明是這個簽名Sign錯了,后來還是安卓的同事提示慢慢試出來了掂林,注意參數(shù)是"appid","noncestr","prepayid"臣缀,package,timestamp泻帮,partnerid精置,這6個,其中partnerid是商戶id,可以讓后臺返回锣杂,也可以在自己申請的微信支付平臺賬號中找到脂倦,直接寫死并且所有的參數(shù)不用加下劃線元莫,不要像官方文檔那樣使用mch_id這種拼接赖阻,注意mch_id用不上,下面就是要將參數(shù)按照從小到大排序(文檔要求的)然后拼接了柒竞,(sign也可以服務(wù)端生成,服務(wù)端調(diào)用統(tǒng)一下單接口后獲取到預(yù)支付id,再進行二次簽名得到我們需要的sign:參考(https://blog.csdn.net/sinat_17820239/article/details/51334181?utm_source=blogxgwz9))因為我是直接從后臺給的數(shù)據(jù)里去取參數(shù)播聪,需要過濾不需要的key,也許把問題弄復(fù)雜了朽基,反正這里要做的就是將這些Key從小到大排序,對應(yīng)的Value也是一樣布隔,讓后按照”key1=value1&key2=value2…“拼接。從小到大排序我用的是swift中的方法
var sortedKeyArray = newKeys.sorted { ( key1 : String, key2 : String) -> Bool in return key1 < key2 }
每個人拼接的方法思路不一樣稼虎,我就不獻丑了衅檀,好了,假設(shè)把這6個Key和對應(yīng)的Value按照URL鍵值對的格式(即key1=value1&key2=value2…)拼接成了stringA霎俩,接著拼接行一個申請得到的weixinKey,這個key在微信支付開發(fā)者賬號里找

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

//1.先自己拼接得到了stringA哀军,假設(shè)是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微信支付就做完了,技術(shù)有限柳击,有做的不對的請指教猿推!下篇記錄關(guān)于Swift支付寶支付的集成

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市捌肴,隨后出現(xiàn)的幾起案子蹬叭,更是在濱河造成了極大的恐慌,老刑警劉巖状知,帶你破解...
    沈念sama閱讀 217,084評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件秽五,死亡現(xiàn)場離奇詭異,居然都是意外死亡饥悴,警方通過查閱死者的電腦和手機坦喘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,623評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來铺坞,“玉大人起宽,你說我怎么就攤上這事〖谜ィ” “怎么了坯沪?”我有些...
    開封第一講書人閱讀 163,450評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長擒滑。 經(jīng)常有香客問我腐晾,道長,這世上最難降的妖魔是什么丐一? 我笑而不...
    開封第一講書人閱讀 58,322評論 1 293
  • 正文 為了忘掉前任藻糖,我火速辦了婚禮,結(jié)果婚禮上库车,老公的妹妹穿的比我還像新娘巨柒。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,370評論 6 390
  • 文/花漫 我一把揭開白布洋满。 她就那樣靜靜地躺著晶乔,像睡著了一般。 火紅的嫁衣襯著肌膚如雪牺勾。 梳的紋絲不亂的頭發(fā)上正罢,一...
    開封第一講書人閱讀 51,274評論 1 300
  • 那天,我揣著相機與錄音驻民,去河邊找鬼翻具。 笑死,一個胖子當(dāng)著我的面吹牛回还,可吹牛的內(nèi)容都是我干的裆泳。 我是一名探鬼主播,決...
    沈念sama閱讀 40,126評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼懦趋,長吁一口氣:“原來是場噩夢啊……” “哼晾虑!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起仅叫,我...
    開封第一講書人閱讀 38,980評論 0 275
  • 序言:老撾萬榮一對情侶失蹤帜篇,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后诫咱,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體笙隙,經(jīng)...
    沈念sama閱讀 45,414評論 1 313
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,599評論 3 334
  • 正文 我和宋清朗相戀三年坎缭,在試婚紗的時候發(fā)現(xiàn)自己被綠了竟痰。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,773評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡掏呼,死狀恐怖坏快,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情憎夷,我是刑警寧澤莽鸿,帶...
    沈念sama閱讀 35,470評論 5 344
  • 正文 年R本政府宣布,位于F島的核電站拾给,受9級特大地震影響祥得,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜蒋得,卻給世界環(huán)境...
    茶點故事閱讀 41,080評論 3 327
  • 文/蒙蒙 一级及、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧额衙,春花似錦饮焦、人聲如沸怕吴。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,713評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽械哟。三九已至,卻和暖如春殿雪,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背锋爪。 一陣腳步聲響...
    開封第一講書人閱讀 32,852評論 1 269
  • 我被黑心中介騙來泰國打工丙曙, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人其骄。 一個月前我還...
    沈念sama閱讀 47,865評論 2 370
  • 正文 我出身青樓亏镰,卻偏偏與公主長得像,于是被迫代替她去往敵國和親拯爽。 傳聞我的和親對象是個殘疾皇子索抓,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,689評論 2 354

推薦閱讀更多精彩內(nèi)容