内购服务
大家都知道做iOS开发本身的收入有三种来源:出售应用、内购和广告。国内用户通常很少直接 购买应用,因此对于开发者而言(特别是个人开发者),内购和广告收入就成了主要的收入来源。内购营销模式,通常软件本身是不收费的,但是要获得某些特权就 必须购买一些道具,而内购的过程是由苹果官方统一来管理的,所以和Game Center一样,在开发内购程序之前要做一些准备工作(下面的准备工作主要是针对真机的,模拟器省略Provisioning Profile配置过程):
-
前四步和Game Center基本完全一致,只是在选择服务时不是选择Game Center而是要选择内购服务(In-App Purchase)。
-
到iTuens Connect中设置“App 内购买项目”,这里仍然以上面的“KCTest”项目为例,假设这个足球竞技游戏中有三种道具,分别为“强力手套”(增强防御)、“金球”(增加金球率) 和“能量瓶”(提供足够体力),前两者是非消耗品只用一次性购买,后者是消耗品用完一次必须再次购买。
-
到iTunes Connect中找到“协议、税务和银行业务”增加“iOS Paid Applications”协议,并完成所有配置后等待审核通过(注意这一步如果不设置在应用程序中无法获得可购买产品)。
-
在 iOS“设置”中找到”iTunes Store与App Store“,在这里可以选择使用沙盒用户登录或者处于注销状态,但是一定注意不能使用真实用户登录,否则下面的购买测试不会成功,因为到目前为止我们的 应用并没有真正通过苹果官方审核只能用沙盒测试用户(如果是模拟器不需要此项设置)。
-
有了上面的设置之后保证应用程序Bundle ID和iTunes Connect中的Bundle ID(或者说App ID中配置的Bundle ID)一致即可准备开发。
开发内购应用时需要使用StoreKit.framework,下面是这个框架中常用的几个类:
SKProduct: 可购买的产品(例如上面设置的能量瓶、强力手套等),其productIdentifier属性对应iTunes Connect中配置的“产品ID“,但是此类不建议直接初始化使用,而是要通过SKProductRequest来加载可用产品(避免出现购买到无效的 产品)。
SKProductRequest:产品请求类,主要用于加载产品列表(包括可用产品和不可用产品),通常加载完之后会通过其 -(void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response代理方法获得响应,拿到响应中的可用产品。
SKPayment:产品购买支付类,保存了产品ID、购买数量等信息(注意与其对应的有一个SKMutablePayment对象,此对象可以修改产品数量等信息)。
SKPaymentQueue: 产品购买支付队列,一旦将一个SKPayment添加到此队列就会向苹果服务器发送请求完成此次交易。注意交易的状态反馈不是通过代理完成的,而是通过一 个交易监听者(类似于代理,可以通过队列的addTransactionObserver来设置)。
SKPaymentTransaction: 一次产品购买交易,通常交易完成后支付队列会调用交易监听者的-(void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transaction方法反馈交易情况,并在此方法中将交易对象返回。
SKStoreProductViewController: 应用程序商店产品展示视图控制器,用于在应用程序内部展示此应用在应用商店的情况。(例如可以使用它让用户在应用内完成评价,注意由于本次演示的示例程序 没有正式提交到应用商店,所以在此暂不演示此控制器视图的使用)。
了解了以上几个常用的开发API之后,下面看一下应用内购买的流程:
-
通 过SKProductRequest获得可购买产品SKProduct数组(SKProductRequest会根据程序的Bundle ID去对应的内购配置中获取指定ID的产品对象),这个过程中需要知道产品标识(必须和iTuens Connect中的对应起来),可以存储到沙盒中也可以存储到数据库中(下面的Demo中定义成了宏定义)。
-
请求完成后 可以在SKProductRequest的-(void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response代理方法中获得SKProductResponse对象,这个对象中保存了products属性表示可用产品对象数组。
-
给 SKPaymentQueue设置一个监听者来获得交易的状态(它类似于一个代理),监听者通过-(void)paymentQueue: (SKPaymentQueue *)queue updatedTransactions:(NSArray *)transaction方法反馈交易的变化状态(通常在此方法中可以根据交易成功、恢复成功等状态来做一些处理)。
-
一 旦用户决定购买某个产品(SKProduct),就可以根据SKProduct来创建一个对应的支付对象SKPayment,只要将这个对象加入到 SKPaymentQueue中就会触发购买行为(将订单提交到苹果服务器),一旦一个交易发生变化就会触发SKPaymentQueue监听者来反馈交 易情况。
-
交易提交给苹果服务器之后如果不出意外的话通常就会弹出一个确认购买的对话框,引导用户完成交易,最终完成交易 后(通常是完成交易,用户点击”好“)会调用交易监听者-(void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transaction方法将此次交易的所有交易对象SKPaymentTransaction数组返回,可以通过交易状态判断交易情况。
-
通 常一次交易完成后需要对本次交易进行验证,避免越狱机器模拟苹果官方的反馈造成交易成功假象。苹果官方提供了一个验证的URL,只要将交易成功后的凭证 (这个凭证从iOS7之后在交易成功会会存储到沙盒中)传递给这个地址就会给出交易状态和本次交易的详细信息,通过这些信息(通常可以根据交易状态、 Bundler ID、ProductID等确认)可以标识出交易是否真正完成。
-
对于非消耗品,用户在完成购买后如果用 户使用其他机器登录或者用户卸载重新安装应用后通常希望这些非消耗品能够恢复(事实上如果不恢复用户再次购买也不会成功)。调用 SKPaymentQueue的restoreCompletedTransactions就可以完成恢复,恢复后会调用交易监听者的 paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transaction方法反馈恢复的交易(也就是已购买的非消耗品交易,注意这个过程中如果没有非消耗品可恢复,是不会调用此方法的)。
下面通过一个示例程序演示内购和恢复的整个过程,程序界面大致如下:
主界面中展示了所有可购买产品和售价,以及购买情况。
选择一个产品点”购买“可以购买此商品,购买完成后刷新购买状态(如果是非消耗品则显示已购买,如果是消耗品则显示购买个数)。
程序卸载后重新安装可以点击”恢复购买“来恢复已购买的非消耗品。
程序代码:
1 //
2 // KCMainTableViewController.m
3 // kctest
4 //
5 // Created by Kenshin Cui on 14/4/5.
6 // Copyright (c) 2015年 cmjstudio. All rights reserved.
7 //
8 #import "KCMainTableViewController.h"
9 #import
10 #define kAppStoreVerifyURL @"https://buy.itunes.apple.com/verifyReceipt" //实际购买验证URL
11 #define kSandboxVerifyURL @"https://sandbox.itunes.apple.com/verifyReceipt" //开发阶段沙盒验证URL
12 //定义可以购买的产品ID,必须和iTunes Connect中设置的一致
13 #define kProductID1 @"ProtectiveGloves" //强力手套,非消耗品
14 #define kProductID2 @"GoldenGlobe" //金球,非消耗品
15 #define kProductID3 @"EnergyBottle" //能量瓶,消耗品
16 @interface KCMainTableViewController ()
17 @property (strong,nonatomic) NSMutableDictionary *products;//有效的产品
18 @property (assign,nonatomic) int selectedRow;//选中行
19 @end
20 @implementation KCMainTableViewController
21 #pragma mark - 控制器视图方法
22 - (void)viewDidLoad {
23 [super viewDidLoad];
24 [self loadProducts];
25 [self addTransactionObjserver];
26 }
27 #pragma mark - UI事件
28 //购买产品
29 - (IBAction)purchaseClick:(UIBarButtonItem *)sender {
30 NSString *productIdentifier=self.products.allKeys[self.selectedRow];
31 SKProduct *product=self.products[productIdentifier];
32 if (product) {
33 [self purchaseProduct:product];
34 }else{
35 NSLog(@"没有可用商品.");
36 }
37
38 }
39 //恢复购买
40 - (IBAction)restorePurchaseClick:(UIBarButtonItem *)sender {
41 [self restoreProduct];
42 }
43 #pragma mark - UITableView数据源方法
44 - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
45 return 1;
46 }
47 - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
48 return self.products.count;
49 }
50 - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
51 static NSString *identtityKey=@"myTableViewCellIdentityKey1";
52 UITableViewCell *cell=[self.tableView dequeueReusableCellWithIdentifier:identtityKey];
53 if(cell==nil){
54 cell=[[UITableViewCell alloc]initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:identtityKey];
55 }
56 cell.accessoryType=UITableViewCellAccessoryNone;
57 NSString *key=self.products.allKeys[indexPath.row];
58 SKProduct *product=self.products[key];
59 NSString *purchaseString;
60 NSUserDefaults *defaults=[NSUserDefaults standardUserDefaults];
61 if ([product.productIdentifier isEqualToString:kProductID3]) {
62 purchaseString=[NSString stringWithFormat:@"已购买%i个",[defaults integerForKey:product.productIdentifier]];
63 }else{
64 if([defaults boolForKey:product.productIdentifier]){
65 purchaseString=@"已购买";
66 }else{
67 purchaseString=@"尚未购买";
68 }
69 }
70 cell.textLabel.text=[NSString stringWithFormat:@"%@(%@)",product.localizedTitle,purchaseString] ;
71 cell.detailTextLabel.text=[NSString stringWithFormat:@"%@",product.price];
72 return cell;
73 }
74 #pragma mark - UITableView代理方法
75 -(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
76 UITableViewCell *currentSelected=[tableView cellForRowAtIndexPath:indexPath];
77 currentSelected.accessoryType=UITableViewCellAccessoryCheckmark;
78 self.selectedRow=indexPath.row;
79 }
80 -(void)tableView:(UITableView *)tableView didDeselectRowAtIndexPath:(NSIndexPath *)indexPath{
81 UITableViewCell *currentSelected=[tableView cellForRowAtIndexPath:indexPath];
82 currentSelected.accessoryType=UITableViewCellAccessoryNone;
83 }
84 #pragma mark - SKProductsRequestd代理方法
85 /**
86 * 产品请求完成后的响应方法
87 *
88 * @param request 请求对象
89 * @param response 响应对象,其中包含产品信息
90 */
91 -(