內購服務數據庫
你們都知道作iOS開發自己的收入有三種來源:出售應用、內購和廣告。國內用戶一般不多直接 購買應用,所以對於開發者而言(特別是我的開發者),內購和廣告收入就成了主要的收入來源。內購營銷模式,一般軟件自己是不收費的,可是要得到某些特權就 必須購買一些道具,而內購的過程是由蘋果官方統一來管理的,因此和Game Center同樣,在開發內購程序以前要作一些準備工做(下面的準備工做主要是針對真機的,模擬器省略Provisioning Profile配置過程):數組
前四步和Game Center基本徹底一致,只是在選擇服務時不是選擇Game Center而是要選擇內購服務(In-App Purchase)。服務器
到iTuens Connect中設置「App 內購買項目」,這裏仍然以上面的「KCTest」項目爲例,假設這個足球競技遊戲中有三種道具,分別爲「強力手套」(加強防護)、「金球」(增長金球率) 和「能量瓶」(提供足夠體力),前二者是非消耗品只用一次性購買,後者是消耗品用完一次必須再次購買。併發
到iTunes Connect中找到「協議、稅務和銀行業務」增長「iOS Paid Applications」協議,並完成全部配置後等待審覈經過(注意這一步若是不設置在應用程序中沒法得到可購買產品)。app
在 iOS「設置」中找到」iTunes Store與App Store「,在這裏能夠選擇使用沙盒用戶登陸或者處於註銷狀態,可是必定注意不能使用真實用戶登陸,不然下面的購買測試不會成功,由於到目前爲止咱們的 應用並無真正經過蘋果官方審覈只能用沙盒測試用戶(若是是模擬器不須要此項設置)。框架
有了上面的設置以後保證應用程序Bundle ID和iTunes Connect中的Bundle ID(或者說App ID中配置的Bundle ID)一致便可準備開發。異步
開發內購應用時須要使用StoreKit.framework,下面是這個框架中經常使用的幾個類:ide
SKProduct: 可購買的產品(例如上面設置的能量瓶、強力手套等),其productIdentifier屬性對應iTunes Connect中配置的「產品ID「,可是此類不建議直接初始化使用,而是要經過SKProductRequest來加載可用產品(避免出現購買到無效的 產品)。測試
SKProductRequest:產品請求類,主要用於加載產品列表(包括可用產品和不可用產品),一般加載完以後會經過其 -(void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response代理方法得到響應,拿到響應中的可用產品。ui
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 -(void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response{ 92 //