iOS-内购的那些事(漏单的问题)
前言
说起内购,其实挺令开发者厌烦的,原因呢,先不说漏单的问题,首先苹果要扣除30%的销售额哦,可恨不?(我觉得可恨),有些想办法先隐藏掉第三方支付(支付宝、微信等),等项目上线了,再跳过内购使用第三方支付,emmmm.......这个方法确实不错,但是如果被苹果发现了,app内虚拟产品调用第三方支付,那好吧,直接下架吧(或许没这么惨,但会惨不忍睹),不要说发现不了,会有人举报哦(别问我怎么知道的);其次就是漏单问题的处理,这一直是个问题,我的项目里虽然做了处理,但是还是会避免不了漏单的,只是把漏单的几率降到了很小,以确保我们维护,给大家分享下内购及漏单的处理。
内购集成
内购集成并不难,这里我用了一个git上封装好的(iaphelper),自己封装也简单(不想造那么多*了),封装的话,建议结合单例封装,充值验证都在单例里面进行;当然也可以不封装直接就用的,这里不多说了。
支付逻辑
1.临时单号
首先根据内购商品id(此商品id是在苹果后台建好的内购商品)、用户信息(后台要求),传给服务器获取一个临时单号,然后先将该临时单号保存到一个变量里。
///获取充值临时单号 - (void)iapgettemorderidwithproductid:(nsstring *)productid{ self.productid = productid; [svprogresshud showwithstatus:@"请稍后..." ]; nsstring *urlstring = @""; [httptools gethttprequesturl:urlstring requestsuccess:^(id repoes, nsurlsessiondatatask *task) { [svprogresshud dismiss]; nsdictionary *dictem = [httptools respoestodic:repoes]; if ([dictem[@"code"] integervalue] == 1) { ///保存临时单号 self.temporaryorderid = dictem[@"data"]; ///发起内购支付 [self iapstartrecharge]; } else{ [svprogresshud showerrorwithstatus:dictem[@"errmsg"]]; [self errorpost:nil]; } } requestfaile:^(nserror *error) { [svprogresshud showerrorwithstatus:[httptools error:error]]; [self errorpost:nil]; }]; }
2.苹果充值
通过商品id调取苹果内购支付,苹果充值成功后,在返回成功的方法里,首先将上一步中的临时单号、用户信息(这里我取userid)、苹果充值成功返回的data,三个参数一起存入本地(我采用数据库存储)后,然后验证服务器充值(如果苹果充值验证失败,不必做任何操作)。
///发起内购支付 - (void)iapstartrecharge{ [svprogresshud showwithstatus:@"请稍后..."]; nsset* dataset = [[nsset alloc] initwithobjects:self.productid, nil]; [iapshare sharedhelper].iap = [[iaphelper alloc] initwithproductidentifiers:dataset]; // 请求商品信息 [[iapshare sharedhelper].iap requestproductswithcompletion:^(skproductsrequest* request,skproductsresponse* response){ if(response.products.count > 0 ) { skproduct *product = response.products[0]; [[iapshare sharedhelper].iap buyproduct:product oncompletion:^(skpaymenttransaction* trans){ if(trans.error){ [svprogresshud showerrorwithstatus:trans.error.userinfo.allvalues[0]]; [self errorpost:nil]; } else if(trans.transactionstate == skpaymenttransactionstatepurchased) { nslog(@"*********内部支付成功*********"); ///将临时单号存在本地【此处做返回信息保存(临时单号、用户信息、返回的data)】 ///去服务器验证充值 } else if(trans.transactionstate == skpaymenttransactionstatefailed) { nslog(@"*********支付失败*********"); if (trans.error.code == skerrorpaymentcancelled) { } else if (trans.error.code == skerrorclientinvalid) { } else if (trans.error.code == skerrorpaymentinvalid) { } else if (trans.error.code == skerrorpaymentnotallowed) { } else if (trans.error.code == skerrorstoreproductnotavailable) { } else{ } [svprogresshud showerrorwithstatus:trans.error.userinfo.allvalues[0]]; [self errorpost:nil]; } }]; }else{ // ..未获取到商品 [svprogresshud showerrorwithstatus:@"暂未获取到商品"]; [self errorpost:nil]; } }]; }
3.服务器验证充值(上一步成功后验证)
在苹果充值成功后,根据充值成功返回的数据data、临时单号、用户信息(后台要求)去服务器验证充值,如果验证成功,将上一步存在本地数据库的数据(临时单号、用户信息(这里我取userid)、苹果充值成功返回的data)删除;如果充值失败,即为漏单,但是已经将验证服务器充值的数据存在了本地数据库,可再次尝试,或者稍候尝试,根据自己的提示操作而定。
///向服务器验证进行充值 - (void)iappayovaliddata:(nsstring *)strreceipt temorder:(nsstring *)temorder{ ///验证充值 [svprogresshud showwithstatus:@"正在为您充值..."]; nsstring *urlsting = @""; ///post data【验证参数】 nsmutabledictionary *dicpost = [nsmutabledictionary dictionary]; [httptools posthttprequesturl:urlsting requestpram:dicpost requestsuccess:^(id respoes) { [svprogresshud dismiss]; nsdictionary *dicvalid = [httptools respoestodic:respoes]; if ([dicvalid[@"code"] integervalue] == 1) {
///删除本地存的验证信息【临时单号、用户信息、苹果支付成功返回的data】 } else{ [self errorpost:dicvalid[@"errmsg"]]; } } requestfaile:^(nserror *erro) { [svprogresshud dismiss]; [self errorpost:[httptools error:erro]]; }]; }
结束
至此,整个内购充值流程已完毕,以上传递的参数、存储的参数,是根据服务器后台要求,可根据自己服务器后台商量,怎么做更好,如果大家有更好的方案,希望能借鉴!谢谢!
最后还是要说说,不管怎么做漏单处理,总会有几个漏单的,但是几率很小,而且还想说,什么时候可以有办法躲过这30%的抽成.......
上一篇: 注释