本文会给大家详细介绍iOS内购,这是本人16年5月底的开发过程,希望对看完此篇文章的人有所帮助。

本文基于XcodeVersion 7.3 (7D175)版本,手机是iPhone 6,9.3系统。

部分地方直接摘自网络,基本上是我的逻辑,省时省心省力。

一. 创建测试App

首先你需要登录 App的ItunesConnection,你会看到如下界面

简单的介绍一下这几个选项

1.我的App主要用于管理自己的App应用,例如编辑资料,上架,下架等。
2.销售和趋势主要是来查看App在各个平台的下载量,收入等方面数据,里面有曲线图等图文结合的方式给我们参考。
3.付款和财务报告显示的是你的收入以及付款等相关信息。
4.iAd主要是跟广告有关,开发者可以登录到Workbench,通过iAd对应用的广告进行控制。
5.用户和职能用于生成相应账号,例如苹果沙河测试账号。
6.协议,税务和银行业务则是你银行相关账户的信息设置。
在这里我们选择第一个选项,我的App, 然后点击左上角的加号,新建一个用来测试用的App。

点新建 App,会出现新建窗口;

在这里有几个需要填写的地方,名称自己取,平台IOS,语言选择了简体中文,套装ID也就是你的Bundle Identifier,需要你在Certificates页面 申请BundleID,SKU可以理解为用户看一看到的唯一标示,会体现在你的app的App Store的链接中。

二.添加内购

App创建好之后,我们打开创建的App,在左上角选择功能,会看到左侧的App 内购买项目。我们点击右下角的加号,为App添加内购项目。

之后我们会看到类型的选项,如下图

官方的注释写的很清楚了,只在这里简单的说下前两种:

- 消耗型项目 就像你玩游戏需要买金币,买钻石等,只要花钱就可以无限次的购买

- 非消耗型项目 就像你在App Store购买App,买了一次之后就不用再买第二次,你拥有永久使用权。

在我们的app中,是充值会员,所以选择的是第一种,可以无限次购买。

这里有几个选项,需要填写商品名称,产品ID以及价格等级,简单说明一下

1. 商品名称根据你的消费道具的实际意义来说明,比如“100颗宝石”,“100金币”等。

2. 产品ID是比较重要的,由项目自定义,只要唯一即可,因为测试,我在这里随便填写的123,在实际应用中,一定要认真填写。

3. 价格等级的话“查看价格表”中有对应的说明,可以对照着表中每个国家的货币价格与等级来选择

接下来是语言选择,和上传快照如下图

点击添加语言,填写名称和描述,这里我们依然选择简体中文,如下

审核备注,根据实际情况填写,可以不填。而下面的屏幕快照,则是商品图片,以像素为单位,最低尺寸为321,390,尺寸需求如下图,上传即可。

到这里为止, 我们的内购项目则添加完成。接下来则是测试阶段了。

三.申请沙盒测试账号(用来测试购买项目)

这个账号,是利用苹果的沙盒测试环境来模拟AppStore的购买流程,你肯定不会想要用真实RMB去购买测试吧?

首先我们回到iTunes Connect中,在这里我们选择用户和职能。

然后在上面的第三个选项沙箱技术测试员中点击加号,添加测试员。

在信息填写页面只简单说两句。

所有信息都可以随意填写,不用管是否真实。

App Store地区选择,一定要选对,它对应的是你创建的App的地区, 你App是中国的话, 在这里我们依然选择中国。

此账号只能用来测试,不要在正式的appstore上使用

填写完毕,点击保存后,我们则生成一个测试账号,当然这个账号是可以随时删除和添加的。

之后终于到了写代码的时候了,点开你的Xcode创建你的项目!

大部分代码都可以在.m文件中实现。

#import "ViewController.h"
#import <StoreKit/StoreKit.h>
#import "SVProgressHUD.h" @interface ViewController ()<SKPaymentTransactionObserver,SKProductsRequestDelegate>
@property (nonatomic,copy) NSString *currentProId; @end @implementation ViewController - (void)viewDidLoad {
   [super viewDidLoad];    UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
   button.frame = CGRectMake(100, 100, 100, 100);
   button.backgroundColor = [UIColor greenColor];
   [button setTitle:@"6元" forState:UIControlStateNormal];
   [button addTarget:self action:@selector(btnClick:) forControlEvents:UIControlEventTouchDown];
   [self.view addSubview:button];
} - (void)btnClick:(UIButton *)button
{
   [[SKPaymentQueue defaultQueue] addTransactionObserver:self];
   _currentProId = @"123";
   if([SKPaymentQueue canMakePayments]){
       [self requestProductData:product];
   }else{
       NSLog(@"不允许程序内付费");
   }
} //去苹果服务器请求商品
- (void)requestProductData:(NSString *)type{
   NSLog(@"-------------请求对应的产品信息----------------");    [SVProgressHUD showWithStatus:nil maskType:SVProgressHUDMaskTypeBlack];    NSArray *product = [[NSArray alloc] initWithObjects:type,nil];    NSSet *nsset = [NSSet setWithArray:product];
   SKProductsRequest *request = [[SKProductsRequest alloc] initWithProductIdentifiers:nsset];
   request.delegate = self;
   [request start]; } //收到产品返回信息
- (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response{    NSLog(@"--------------收到产品反馈消息---------------------");
   NSArray *product = response.products;
   if([product count] == 0){
       [SVProgressHUD dismiss];
       NSLog(@"--------------没有商品------------------");
       return;
   }    NSLog(@"productID:%@", response.invalidProductIdentifiers);
   NSLog(@"产品付费数量:%lu",(unsigned long)[product count]);    SKProduct *p = nil;
   for (SKProduct *pro in product) {
       NSLog(@"%@", [pro description]);
       NSLog(@"%@", [pro localizedTitle]);
       NSLog(@"%@", [pro localizedDescription]);
       NSLog(@"%@", [pro price]);
       NSLog(@"%@", [pro productIdentifier]);        if([pro.productIdentifier isEqualToString:_currentProId]){
           p = pro;
       }
   }    SKPayment *payment = [SKPayment paymentWithProduct:p];    NSLog(@"发送购买请求");
   [[SKPaymentQueue defaultQueue] addPayment:payment];
} //请求失败
- (void)request:(SKRequest *)request didFailWithError:(NSError *)error{
   [SVProgressHUD showErrorWithStatus:@"支付失败"];
   NSLog(@"------------------错误-----------------:%@", error);
} - (void)requestDidFinish:(SKRequest *)request{
   [SVProgressHUD dismiss];
   NSLog(@"------------反馈信息结束-----------------");
}
//沙盒测试环境验证
#define SANDBOX @"https://sandbox.itunes.apple.com/verifyReceipt"
//正式环境验证
#define AppStore @"https://buy.itunes.apple.com/verifyReceipt"
/**
*  验证购买,避免越狱软件模拟苹果请求达到非法购买问题
*
*/
-(void)verifyPurchaseWithPaymentTransaction{
   //从沙盒中获取交易凭证并且拼接成请求体数据
   NSURL *receiptUrl=[[NSBundle mainBundle] appStoreReceiptURL];
   NSData *receiptData=[NSData dataWithContentsOfURL:receiptUrl];    NSString *receiptString=[receiptData base64EncodedStringWithOptions:NSDataBase64EncodingEndLineWithLineFeed];//转化为base64字符串    NSString *bodyString = [NSString stringWithFormat:@"{\"receipt-data\" : \"%@\"}", receiptString];//拼接请求数据
   NSData *bodyData = [bodyString dataUsingEncoding:NSUTF8StringEncoding];    //创建请求到苹果官方进行购买验证
   NSURL *url=[NSURL URLWithString:SANDBOX];
   NSMutableURLRequest *requestM=[NSMutableURLRequest requestWithURL:url];
   requestM.HTTPBody=bodyData;
   requestM.HTTPMethod=@"POST";
   //创建连接并发送同步请求
   NSError *error=nil;
   NSData *responseData=[NSURLConnection sendSynchronousRequest:requestM returningResponse:nil error:&error];
   if (error) {
       NSLog(@"验证购买过程中发生错误,错误信息:%@",error.localizedDescription);
       return;
   }
   NSDictionary *dic=[NSJSONSerialization JSONObjectWithData:responseData options:NSJSONReadingAllowFragments error:nil];
   NSLog(@"%@",dic);
   if([dic[@"status"] intValue]==0){
       NSLog(@"购买成功!");
       NSDictionary *dicReceipt= dic[@"receipt"];
       NSDictionary *dicInApp=[dicReceipt[@"in_app"] firstObject];
       NSString *productIdentifier= dicInApp[@"product_id"];//读取产品标识
       //如果是消耗品则记录购买数量,非消耗品则记录是否购买过
       NSUserDefaults *defaults=[NSUserDefaults standardUserDefaults];
       if ([productIdentifier isEqualToString:@"123"]) {
           int purchasedCount=[defaults integerForKey:productIdentifier];//已购买数量
           [[NSUserDefaults standardUserDefaults] setInteger:(purchasedCount+1) forKey:productIdentifier];
       }else{
           [defaults setBool:YES forKey:productIdentifier];
       }
       //在此处对购买记录进行存储,可以存储到开发商的服务器端
   }else{
       NSLog(@"购买失败,未通过验证!");
   }
}
//监听购买结果
- (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transaction{
   for(SKPaymentTransaction *tran in transaction){
       switch (tran.transactionState) {
           case SKPaymentTransactionStatePurchased:{
               NSLog(@"交易完成");
               // 发送到苹果服务器验证凭证
               [self verifyPurchaseWithPaymentTransaction];
               [[SKPaymentQueue defaultQueue] finishTransaction:tran];            }
               break;
           case SKPaymentTransactionStatePurchasing:
               NSLog(@"商品添加进列表");                break;
           case SKPaymentTransactionStateRestored:{
               NSLog(@"已经购买过商品");                [[SKPaymentQueue defaultQueue] finishTransaction:tran];
           }
               break;
           case SKPaymentTransactionStateFailed:{
               NSLog(@"交易失败");
               [[SKPaymentQueue defaultQueue] finishTransaction:tran];
               [SVProgressHUD showErrorWithStatus:@"购买失败"];
           }
               break;
           default:
               break;
       }
   }
} //交易结束
- (void)completeTransaction:(SKPaymentTransaction *)transaction{
   NSLog(@"交易结束");    [[SKPaymentQueue defaultQueue] finishTransaction:transaction];
} - (void)dealloc{
   [[SKPaymentQueue defaultQueue] removeTransactionObserver:self];
} - (void)didReceiveMemoryWarning {
   [super didReceiveMemoryWarning];
   // Dispose of any resources that can be recreated.
} @end

在这里需要注意几点,

  1. 代码中的_currentProId所填写的是你的购买项目的的ID,这个和第二步创建的内购的productID要一致;本例中是 123。

  2. 在监听购买结果后,一定要调用[[SKPaymentQueue defaultQueue] finishTransaction:tran];来允许你从支付队列中移除交易。

  3. 沙盒环境测试appStore内购流程的时候,请使用没越狱的设备。

  4. 请务必使用真机来测试,一切以真机为准。

  5. 项目的Bundle identifier需要与您申请AppID时填写的bundleID一致,不然会无法请求到商品信息。

  6. 真机测试的时候,一定要退出原来的账号,才能用沙盒测试账号

  7. 二次验证,请注意区分宏, 测试用沙盒验证,App Store审核的时候也使用的是沙盒购买,所以验证购买凭证的时候需要判断返回Status Code决定是否去沙盒进行二次验证,为了线上用户的使用,验证的顺序肯定是先验证正式环境,此时若返回值为21007,就需要去沙盒二次验证,因为此购买的是在沙盒进行的。

附:苹果支付错误目录

iOS应用内支付(内购)的个人开发过程及坑!的更多相关文章

  1. [IPA]IOS In App Purchase(内购)验证

    参考我之前的笔记 苹果内购笔记,在客户端向苹果购买成功之后,我们需要进行二次验证. 二次验证 IOS在沙箱环境下购买成功之后,向苹果进行二次验证,确认用户是否购买成功. 当应用向Apple服务器请求购 ...

  2. iOS: 实现苹果的内购

    一.介绍: 在个人开发的app上架到AppStore后,苹果官方允许我们将自己的app在appstore上进行付费使用,也就是所谓的内购.其中,支付方式规定的必须是苹果的支付方式:应用内支付. 二.流 ...

  3. SDK接入(3)之iOS内支付(In-App Purchase)接入

    SDK接入(3)之iOS内支付(In-App Purchase)接入 继整理了Android平台的SDK接入过程.再来分享下iOS平台的内支付(In-App Purchase)接入,作为笔者在游戏开发 ...

  4. iOS应用内支付(IAP)的那些坑

    本文转载至 http://blog.devtang.com/2013/04/07/tricks-in-iap/ 前言 udacity 中的在线课程 <How to build a startup ...

  5. Cocos2dx使用ios内支付IAP具体流程-白白

    今天总结了一下cocos2d-x使用ios内支付iap的具体流程,封装好了调用接口,代码与具体说明在此 http://download.csdn.net/detail/u010229677/81566 ...

  6. IOS IAP APP内支付 Java服务端代码

    IOS IAP APP内支付 Java服务端代码   场景:作为后台需要为app提供服务,在ios中,app内进行支付购买时需要进行二次验证. 基础:可以参考上一篇转载的博文In-App Purcha ...

  7. iOS内购(IAP)中的那些坑

    公司的公共库原来并没有这部分的代码,以前做内购是用两个比较有名的github上的第三方库.一个叫MKStoreKit,另一个叫IAPManager,我看了一下写的都很辣鸡,使用起来很不方便,而且写的还 ...

  8. SDK接入(2)之Android Google Play内支付(in-app Billing)接入

    SDK接入(2)之Android Google Play内支付(in-app Billing)接入 继上篇SDK接入(1)之Android Facebook SDK接入整理完Facebook接入流程之 ...

  9. 【读书笔记】iOS-验证应用内支付的凭证注意事项

    1,简单来说,越狱后的手机由于没有沙盒作为保护,黑客可以对系统进行任意的修改,所以,在支付过程中,苹果返回的已付款成功的凭证可能是伪造的.客户端拿到付款凭证之后,还需要将凭证上传到自己的服务器,进行二 ...

随机推荐

  1. nginx 一般配置实例 静态页面

    # 使用的用户和组 user www www; # 指定工作衍生进程数(一般等于CPU的总核数或总核数的两倍,例如两个四核CPU,则总核数为8) worker_processes 8; # 指定错误日 ...

  2. Notepad++ Shortcuts(Chinese and English Version)

    Ctrl+C 复制Ctrl+X 剪切Ctrl+V 粘贴Ctrl+Z 撤消Ctrl+Y 恢复Ctrl+A 全选Ctrl+F 键查找对话框启动Ctrl+H 查找/替换对话框Ctrl+D 复制并粘贴当行 C ...

  3. DevExpress::XtraBars::BarEditItem获取EditValue值事件

    //视图设计器中拖动一个barManager,添加一个bar,再添加一个BarEditItem控件,如下代码: private: DevExpress::XtraEditors::Repository ...

  4. STM32学习笔记——定时器中断(向原子哥学习)

    定时器中断 STM32 的定时器功能十分强大,有 TIME1 和 TIME8 等高级定时器,也有 TIME2~TIME5 等通用定时器,还有 TIME6 和TIME7 等基本定时器.在本章中,我们将利 ...

  5. centos下安装chdmg

    http://www.aboutyun.com/thread-9075-1-1.html  基本参考这个   yum clean all yum update     1.保证selinux关闭  / ...

  6. hdu 5151 Sit sit sit

    http://acm.hdu.edu.cn/showproblem.php?pid=5151 题意:一共有N个椅子,然后有N个学生依次去坐,满足下面三个条件不能坐上去,1:这个椅子旁边有左椅子也有右椅 ...

  7. ios入门之c语言篇——基本函数——1——随机数生成

    1.随机数函数 参数返回值解析: 参数: a:int,数字范围最小值: b:int,数字范围最大值: 返回值: 1:闰年: 0:非闰年: 备注: a-b的绝对值不能超过int的最大值(65535); ...

  8. Altium Designer 从导入DXF文件,并转换成板框

    大多数人都知道,PADS中导入DXF文件,然后转换成板框,是很方便的.AD也同样可以做到. PADS导入DXF见:http://www.cnblogs.com/craftor/archive/2012 ...

  9. fedora下体验gentoo安装

    服务器上安装了fedora,但是对gentoo很想体验一番,没有新机器,不想重装系统,所以只能chroot来体验getoo了. 下载portage-20130817.tar.bz2和stage3-am ...

  10. 【Linux】鸟哥的Linux私房菜基础学习篇整理(八)

    1. useradd [-u UID] [-g 初始用户组] [-G 次要用户组] [-mM]\   [-c 说明栏] [-d 主文件夹绝对路径] [-r] [-s shell] 用户账号名:新增用户 ...