iOS微信支付填坑记

作者: 黑暗中的孤影 | 来源:发表于2016-06-04 15:12 被阅读3520次

    最近公司的APP需要接入微信支付功能。因为在这之前做了不少功课,所以总体步骤还是比较顺利,但是最后在拉起微信App里,总是出现-2错误码。这个地方坑了我好长时间,我相信读者可能会砬到过这种情况。最后在安卓同事的帮助下总算搞定了,成功进入支付页面。下面我来总结iOS微信支付的坑。

    1: 下载微信支付SDK和Demo,网址微信支付

    在这里我不得不吐槽微信支付的Demo开发者,此Demo还是2012年写的,是用MRC来管理内存,而且这个Demo我一直没有运行起来。我个人建议不要看这个Demo了,按照我下面的说明配置即可。下载完成后将SDK文件解压后放到你的项目里。在这里我说明一下,如果你用了友盟的社会化分享并且还支持微信的话,那你就不需要下载微信支付SDK了,直接可以用友盟里面的SDK文件。

    2:配置App跳转的白名单。iOS9 大家都懂的

    App跳转白名单

    3:配置iOS UrlSchema,这个作为iOS开发者都知道啦

    Url Schema 就是你在申请微信支付的App key


    配置URL Types

    4:添加框架Framework和dylib文件

    有时侯添加SDK后项目不能编译通过,这种通常是你还没有加入SDK必要的dylib或者framework文件,确保在Link Binary With Libraries里加入以下Library


    添加需要的Library
    SystemConfiguration.framework,
    libz.dylib,
    libsqlite3.0.dylib,
    libc++.dylib
    CoreTelephony.frame
    

    5 :在AppDelegate文件里面的方法里面微信注册您的APPID

    - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
        // Override point for customization after application launch.
        
        [WXApi registerApp:@"此处填入微信分配的APPID"];//一般写以wx开头的Id
        //[WXApi registerApp:@"wxd930ea5d5a258f4f" withDescription:@"demo 2.0"];
        return YES;
    }
    

    里面的两个方法使用做任意一个即可

    6:获取订单的重要信息,这个一般Server的同事会提供。我们需要下面最5个关键的属性:

     "appid": "wxb423c568326b71ee",//微信开放平台审核通过的应用APPID
     "partnerId": "1326631501",//微信支付分配的商户号
     "nonceStr": "EStCEnC8lVvIBV10",//随机字符串
     "prepayId": "wx20160601113412a39d0f4d700072397236",//预支付交易会话标识
     "sign":"E0A8028B79F4E64885B9D930E6FD88E2" //sign,这个是签名,但是很有可能这个签名是有问题的,我用不了
    

    7:生成PayReq对象

    PayReq *request = [PayReq new];
    request.partnerId = @"1326631501";
    request.prepayId= @"wx20160601113412a39d0f4d700072397236";
    request.package = @"Sign=WXPay"; //package必定是这个
    request.nonceStr= @"EStCEnC8lVvIBV10";
     NSDate *datenow = [NSDate date];
          NSString *timeSp = [NSString stringWithFormat:@"%ld", (long)[datenow timeIntervalSince1970]];
       UInt32 timeStamp =[timeSp intValue];
       request.timeStamp= timeStamp; //timeStamp是个十位数字
    
    //request.sign= @"E0A8028B79F4E64885B9D930E6FD88E2";
       DataMD5 *md5 = [[DataMD5 alloc] init];
            request.sign=[md5 createMD5SingForPay:payInfo.appId partnerid:p.partnerId prepayid:p.prepayId package:p.package noncestr:p.nonceStr timestamp:p.timeStamp];
            NSLog(@"%@",p.sign);
           
    [WXApi sendReq:request];
    

    在这里值得说明;对于timeStamp,对当前的时间戳就行。再将其转成10位数字。
    对于sign,在这里有个大坑,如果你直接使用了server返回的sign,那么在调起微信App后,发现这个页面只有一个确定按钮,无法实现支付功能,

    调起微信支付后出现这种情况

    点了确定按钮返回App后,给出的错误码中-2,就是是说,用户取消支付,什么玩意?
    这其实就是传递接口的字段导致的问题。那么如何解决呢?首先确保以下几点

    • nonceStr是用Server返回的,不要自己生成。
    • request.package = @"Sign=WXPay", 这个是死的,不要写错。
    • timeStamp是个十位数字,不是13位。
    • 确定前面所有都没有错误后,很有可能就是sign的问题了。那么我们需要自己完成签名,也就是说自己生成sign。
      生成Sign需要DataMD5类,建议读者自己下载 WebchatDemo里面有这个类
      所传的参数appId要用Server提供的appId,不要弄错成其他的了,注意这里还有一个坑,在DataMD5的m文件里面要添加商户密钥key字段
    要注意在加密的方法里添加商户密钥key
      [contentString appendFormat:@"key=%@",@"E0A8028B79F4E64885B9D930E6FD88E2"];
    
    • 最后就是sign是全部大写了
    • 还有最后一个问题,就是如果你用了友盟社会化分享来做微信登录的话,那么你要先用友盟注册微信,后调用微信SDK注册
          [UMSocialWechatHandler setWXAppId:SHARE_WX appSecret:SHARE_WX_SECRET url:SHAREURL];
            
          [WXApi registerApp:SHARE_WX withDescription:@"demo"];
    

    这下返回码为-2的问题应该可以解决了,如果还是不行,可以和Server的同事商量返回的属性有没有问题。

    8:最后还的回调问题,当用户完成支付或者取消支付,都要跳转回App,需要在这里处理一下url

     if url.host?.compare("pay") == NSComparisonResult.OrderedSame && url.host == "pay" {
             WXApi.handleOpenURL(url,delegate: WXApiManager.sharedManager())
             return true
            }
    

    如果返回的url是pay,那么就是微信支付的跳回。需要调用 WXApi.handleOpenURL来处理,需要把Demo的WXApiManager文件拷贝到项目里面,调用它来处理回调。

    上面就是我在开发微信支付SDK遇到的坑了,希望可以帮助大家。

    相关文章

      网友评论

      • Ko_Neko:博主你好,请问一下在回调里是如何判断支付是成功还是失败的?
      • Schorem:你好 你碰到过友盟分享与微信支付回调冲突问题吗? 微信支付完经过友盟分享的回调判断后返回yes导致不走支付的回调
      • 心底碎片:提示签名失败
      • 不余先生:甲由甲由
      • 孙公瑾i:“最后就是sign是全部大写了” 这句话什么个意思?我一只返回支付签名失败是咋回事,retcode=-2
      • ac2c5e86bd8c:感谢分享
      • kCornerRadius:博主说的商户密钥是指WX_APP_SECRET这种吗?我看我们的都是小写,文章里边的是大写的
        黑暗中的孤影:@kCornerRadius 如果你可以正常支付的话,那就不会有影响的
      • Pusswzy:无法修改APPID, 无论我怎么修改, 都是微信demo里面的wxb4ba3c02aa476ea1
        Pusswzy:@黑暗中的孤影 当时2了 没有更换二次签名的接口 可是还是只有确定按钮, 估计需要自己生成签名了
        黑暗中的孤影:@Pusswzy AppID是申请微信支付时微信给的的一个id。这个是要用自己的,我不明白你在是什么意思。
      • mark666:你写的是后台加密吧?
        黑暗中的孤影:@mark666 不是后台加密
      • DevImp:我遇到了跟你一样的问题 用你的方法解决了 但是商户密钥放在客户端是件很不安全的事。
        我检查了一下代码发现是填充PayReq.timeStamp = timestamp 我这个之前是untiy3D http请求传过来的NSString 我没有做[timestamp intValue]所以第一次失败是用服务器传过来的正确的签名+错误的时间戳 第二次成功是因为用了错误的时间戳+但是我用错误的时间戳客户端本地生成了签名这样就是成功的。XCode 这种UInt32 = NString* 这种为什么警告都没有 而且就算隐形转化 也转化的不对 可能是把NString的地址转的过去吧。
      • a4a8de24c022:字字珠玑,受益匪浅 :+1:
      • 剑剑龙:是的,非常坑
      • qqzysh:楼主, 你的 prepayId 是如何得到的, 谢谢楼主, 希望能告知 :blush:
        qqzysh:@黑暗中的孤影 Thanks
        黑暗中的孤影:@qqzysh 这个是服务器的同事提供的
      • 姜朋升2:楼主,其实根本就没有坑,咱们需要做的就是把官方提供的那个接口换成咱们自己的就行
        黑暗中的孤影: @姜朋升1 那你们server的人的签名做对了。我们的不行
        姜朋升2:我们是可以用的
        黑暗中的孤影:@姜朋升1 你确定你们服务器提供的sign可以用,如果可以用,那完全没问题。但是很多人是用不了的,需要自己手动sign。
      • 1b86d918449d:新技能get
      • BestJoker:很不错,加油

      本文标题:iOS微信支付填坑记

      本文链接:https://www.haomeiwen.com/subject/evmldttx.html