iOS應(yīng)用內(nèi)付費(fèi)(IAP)開發(fā)步驟列表
前兩天和服務(wù)端同事一起,完成了應(yīng)用內(nèi)付費(fèi)(以下簡(jiǎn)稱IAP, In app purchase)的開發(fā)工作。步驟繁多,在此把開發(fā)步驟列表整理如下。因?yàn)橹皇遣襟E列表,所以并不含詳細(xì)的說明教程,需要看教程的新手,可以看我附在最后的一些參考鏈接。
配置Developer.apple.com
登錄到Developer.apple.com,然后進(jìn)行以下步驟:
- 為應(yīng)用建立建立一個(gè)不帶通配符的App ID
- 用該App ID生成和安裝相應(yīng)的Provisioning Profile文件。
配置iTunes Connect
登錄到iTunes Connet,然后進(jìn)行以下步驟:
1.用該App ID創(chuàng)建一個(gè)新的應(yīng)用。
2.在該應(yīng)用中,創(chuàng)建應(yīng)用內(nèi)付費(fèi)項(xiàng)目,選擇付費(fèi)類型,通??蛇x的是可重復(fù)消費(fèi)(Consumable)的或是永久有效(Non-Consumable)的2 種,然后設(shè)置好價(jià)格和Product ID以及購買介紹和截圖即可,這里的Product ID是需要記住的,后面開發(fā)的時(shí)候需要。如下圖所示:
3.添加一個(gè)用于在sandbox付費(fèi)的測(cè)試用戶,如下圖所示。注意蘋果對(duì)該測(cè)試用戶的密碼要求 和正式賬號(hào)一樣,必須是至少8位,并且同時(shí)包含數(shù)字和大小寫字母:
4.填寫相關(guān)的稅務(wù),銀行,聯(lián)系人信息。如下圖所示:
開發(fā)工作(ios端)
1、 在工程中引入 storekit.framework 和 #import <StoreKit/StoreKit.h>
2、 獲得所有的付費(fèi)Product ID列表。這個(gè)可以用常量存儲(chǔ)在本地,也可以由自己的服務(wù)器返回。
3、 制作一個(gè)界面,展示所有的應(yīng)用內(nèi)付費(fèi)項(xiàng)目。這些應(yīng)用內(nèi)付費(fèi)項(xiàng)目的價(jià)格和介紹信息可以是自己的服務(wù)器返回。但如果是不帶服務(wù)器的單機(jī)游戲應(yīng)用或工具類應(yīng)用, 則可以通過向App Store查詢獲得。我在測(cè)試時(shí)發(fā)現(xiàn),向App Store查詢速度非常慢,通常需要2-3秒鐘,所以不建議這么做,最好還是搞個(gè)自己的服務(wù)器吧。
4、當(dāng)用戶點(diǎn)擊了一個(gè)IAP項(xiàng)目,我們先查詢用戶是否允許應(yīng)用內(nèi)付費(fèi),如果不允許則不用進(jìn)行以下步驟了。代碼如下:
- if ([SKPaymentQueue canMakePayments]) {
- // 執(zhí)行下面提到的第5步:
- [self getProductInfo];
- } else {
- NSLog(@"失敗,用戶禁止應(yīng)用內(nèi)付費(fèi)購買.");
- }
5、 我們先通過該IAP的ProductID向AppStore查詢,獲得SKPayment實(shí)例,然后通過SKPaymentQueue的 addPayment方法發(fā)起一個(gè)購買的操作。
- // 下面的ProductId應(yīng)該是事先在itunesConnect中添加好的,已存在的付費(fèi)項(xiàng)目。否則查詢會(huì)失敗。
- - (void)getProductInfo {
- NSSet * set = [NSSet setWithArray:@[@"ProductId"]];
- SKProductsRequest * request = [[SKProductsRequest alloc] initWithProductIdentifiers:set];
- request.delegate = self;
- [request start];
- }
- // 以上查詢的回調(diào)函數(shù)
- - (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response {
- NSArray *myProduct = response.products;
- if (myProduct.count == 0) {
- NSLog(@"無法獲取產(chǎn)品信息,購買失敗。");
- return;
- }
- SKPayment * payment = [SKPayment paymentWithProduct:myProduct[0]];
- [[SKPaymentQueue defaultQueue] addPayment:payment];
- }
6、 在viewDidLoad方法中,將購買頁面設(shè)置成購買的Observer。
- - (void)viewDidLoad {
- [super viewDidLoad];
- // 監(jiān)聽購買結(jié)果
- [[SKPaymentQueue defaultQueue] addTransactionObserver:self];
- }
- - (void)viewDidUnload {
- [super viewDidUnload];
- [[SKPaymentQueue defaultQueue] removeTransactionObserver:self];
- }
7、 當(dāng)用戶購買的操作有結(jié)果時(shí),就會(huì)觸發(fā)下面的回調(diào)函數(shù),相應(yīng)進(jìn)行處理即可。
- - (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transactions {
- for (SKPaymentTransaction *transaction in transactions)
- {
- switch (transaction.transactionState)
- {
- case SKPaymentTransactionStatePurchased://交易完成
- NSLog(@"transactionIdentifier = %@", transaction.transactionIdentifier);
- [self completeTransaction:transaction];
- break;
- case SKPaymentTransactionStateFailed://交易失敗
- [self failedTransaction:transaction];
- break;
- case SKPaymentTransactionStateRestored://已經(jīng)購買過該商品
- [self restoreTransaction:transaction];
- break;
- case SKPaymentTransactionStatePurchasing: //商品添加進(jìn)列表
- NSLog(@"商品添加進(jìn)列表");
- break;
- default:
- break;
- }
- }
- }
- - (void)completeTransaction:(SKPaymentTransaction *)transaction {
- // Your application should implement these two methods.
- NSString * productIdentifier = transaction.payment.productIdentifier;
- NSString * receipt = [transaction.transactionReceipt base64EncodedString];
- if ([productIdentifier length] > 0) {
- // 向自己的服務(wù)器驗(yàn)證購買憑證
- }
- // Remove the transaction from the payment queue.
- [[SKPaymentQueue defaultQueue] finishTransaction: transaction];
- }
- - (void)failedTransaction:(SKPaymentTransaction *)transaction {
- if(transaction.error.code != SKErrorPaymentCancelled) {
- NSLog(@"購買失敗");
- } else {
- NSLog(@"用戶取消交易");
- }
- [[SKPaymentQueue defaultQueue] finishTransaction: transaction];
- }
- - (void)restoreTransaction:(SKPaymentTransaction *)transaction {
- // 對(duì)于已購商品,處理恢復(fù)購買的邏輯
- [[SKPaymentQueue defaultQueue] finishTransaction: transaction];
- }
8、服務(wù)器驗(yàn)證憑證(Optional)。如果購買成功,我們需要將憑證發(fā)送到服務(wù)器上進(jìn)行驗(yàn)證??紤]到網(wǎng)絡(luò)異常情況,iOS端的發(fā)送憑證操作應(yīng)該進(jìn)行持久化,如果程序退出,崩潰或網(wǎng)絡(luò)異常,可以恢復(fù)重試。
開發(fā)工作(服務(wù)端)
服務(wù)端的工作比較簡(jiǎn)單,分4步:
- 接收ios端發(fā)過來的購買憑證。
- 判斷憑證是否已經(jīng)存在或驗(yàn)證過,然后存儲(chǔ)該憑證。
- 將該憑證發(fā)送到蘋果的服務(wù)器驗(yàn)證,并將驗(yàn)證結(jié)果返回給客戶端。
- 如果需要,修改用戶相應(yīng)的會(huì)員權(quán)限。
考慮到網(wǎng)絡(luò)異常情況,服務(wù)器的驗(yàn)證應(yīng)該是一個(gè)可恢復(fù)的隊(duì)列,如果網(wǎng)絡(luò)失敗了,應(yīng)該進(jìn)行重試。
與蘋果的驗(yàn)證接口文檔在這里。簡(jiǎn)單來說就是將該購買憑證用Base64編碼,然后POST給蘋果的驗(yàn)證服務(wù)器,蘋果將驗(yàn)證結(jié)果以JSON形式返回。
蘋果AppStore線上的購買憑證驗(yàn)證地址是https://buy.itunes.apple.com/verifyReceipt ,測(cè)試的驗(yàn)證地址是:https://sandbox.itunes.apple.com/verifyReceipt
參考鏈接
以下參考鏈接詳細(xì)說明了完成應(yīng)用內(nèi)付費(fèi)開發(fā)的步驟: