重點(diǎn)總結(jié):
1.獲取內(nèi)購列表(從App內(nèi)讀取或從自己服務(wù)器讀认趵谩)
2.App Store請(qǐng)求可用的內(nèi)購列表
3.向用戶展示內(nèi)購列表
4.用戶選擇了內(nèi)購列表,再發(fā)個(gè)購買請(qǐng)求和簸,收到購買完成的回調(diào)(購買完成后會(huì)把錢打給申請(qǐng)內(nèi)購的銀行卡內(nèi))
5.購買流程結(jié)束后, 向服務(wù)器發(fā)起驗(yàn)證憑證以及支付結(jié)果的請(qǐng)求
6.自己的服務(wù)器將支付結(jié)果信息返回給前端并發(fā)放虛擬產(chǎn)品
7.服務(wù)端的工作比較簡(jiǎn)單彭雾,分4步:
7.1.接收ios端發(fā)過來的購買憑證。
7.2.判斷憑證是否已經(jīng)存在或驗(yàn)證過比搭,然后存儲(chǔ)該憑證冠跷。
7.3.將該憑證發(fā)送到蘋果的服務(wù)器驗(yàn)證,并將驗(yàn)證結(jié)果返回給客戶端身诺。
7.4.如果需要蜜托,修改用戶相應(yīng)的會(huì)員權(quán)限。
7.5.考慮到網(wǎng)絡(luò)異常情況霉赡,服務(wù)器的驗(yàn)證應(yīng)該是一個(gè)可恢復(fù)的隊(duì)列橄务,如果網(wǎng)絡(luò)失敗了,應(yīng)該進(jìn)行重試穴亏。
簡(jiǎn)單來說就是將該購買憑證用Base64編碼蜂挪,然后POST給蘋果的驗(yàn)證服務(wù)器,蘋果將驗(yàn)證結(jié)果以JSON形式返回嗓化。
一棠涮、使用注意事項(xiàng)及遇到的坑
####1.使用注意
- 代碼中的_currentProId所填寫的是你的購買項(xiàng)目的的ID,這個(gè)和第二步創(chuàng)建的內(nèi)購的productID要一致刺覆,產(chǎn)品id與_currentProId一致严肪。
- 在監(jiān)聽購買結(jié)果后,一定要調(diào)用[[SKPaymentQueue defaultQueue] finishTransaction:tran];來允許你從支付隊(duì)列中移除交易。
- 真機(jī)測(cè)試的時(shí)候驳糯,一定要退出原來的賬號(hào)(app store 登錄的賬號(hào)退出)篇梭,才能用沙盒測(cè)試賬號(hào)。
- 請(qǐng)務(wù)必使用真機(jī)來測(cè)試酝枢,一切以真機(jī)為準(zhǔn)恬偷。
- 項(xiàng)目的Bundle identifier需要與您申請(qǐng)AppID時(shí)填寫的bundleID一致,不然會(huì)無法請(qǐng)求到商品信息帘睦。
- 沙盒環(huán)境測(cè)試appStore內(nèi)購流程的時(shí)候袍患,請(qǐng)使用沒越獄的設(shè)備。
- 二次驗(yàn)證官脓,請(qǐng)注意區(qū)分宏协怒, 測(cè)試用沙盒驗(yàn)證,App Store審核的時(shí)候也使用的是沙盒購買卑笨,所以驗(yàn)證購買憑證的時(shí)候需要判斷返回Status Code決定是否去沙盒進(jìn)行二次驗(yàn)證孕暇,為了線上用戶的使用,驗(yàn)證的順序肯定是先驗(yàn)證正式環(huán)境赤兴,此時(shí)若返回值為21007妖滔,就需要去沙盒二次驗(yàn)證,因?yàn)榇速徺I的是在沙盒進(jìn)行的桶良。
8.貨幣類型(Bank Account Currency) :填CNY(如果你的app在中國使用的話)座舍。
####2.獲取不到商品信息
1.確定配置環(huán)節(jié)正確。
2.確定是真機(jī)測(cè)試且手機(jī)沒有越獄陨帆。
3.確定內(nèi)購商品添加到了需要內(nèi)購功能的App中曲秉。
4.確定當(dāng)前運(yùn)行的App的Bundle ID和后臺(tái)配置的App的Bundle ID是一致的。
5.可以嘗試先刪除舊App疲牵,再重新編譯生成新的承二,避免新App未覆蓋錯(cuò)誤。
6.這里要提一點(diǎn)纲爸,沙盒的測(cè)試賬號(hào)和你請(qǐng)求商品信息沒有關(guān)系亥鸠。請(qǐng)求商品信息的流程是,你在后臺(tái)配置好了內(nèi)購商品识啦,并且將其添加到了需要集成內(nèi)購功能的App中负蚊,然后你請(qǐng)求商品。請(qǐng)求到商品后的流程是這樣的颓哮,蘋果系統(tǒng)會(huì)自動(dòng)彈出登錄框讓你登錄賬號(hào)家妆。然后根據(jù)提示操作進(jìn)行購買,這里的賬號(hào)就是你配置的沙盒測(cè)試賬號(hào)冕茅。
二伤极、為什么要使用內(nèi)購腰鬼?(why)和內(nèi)購是什么?(what)
1.如果你購買的商品塑荒,是在本app中使用和消耗的,就一定要用內(nèi)購姜挺,否則會(huì)被拒絕上線齿税,例如:游戲幣,在線書籍
app中使用的道具等炊豪。本例中凌箕,就是直播中你用來打賞用的金幣,那東西可就屬于消耗型的词渤。
2.如果是直接購買商城之類的快遞包郵的那些東東牵舱,那就直接調(diào)用支付寶,微信啦缺虐,之類的三方支付就好了芜壁,淘寶,京東都玩過哈高氮!
比較坑的一點(diǎn)就是慧妄,內(nèi)購的話,還要和蘋果3/7分成剪芍,那就可以說塞淹,充值相同的錢,相對(duì)來說罪裹,iOS是比安卓虧的饱普!
三、怎樣使用內(nèi)購状共?(how)
(1)使用內(nèi)購需要哪些資料
1張visa銀行卡套耕,appid,1張銀行卡與蘋果三七分打錢用
協(xié)議口芍、稅務(wù)和銀行業(yè)務(wù)
聯(lián)系人信息:(appid賬號(hào)人)姓名箍铲,郵箱,電話號(hào)碼鬓椭,地址(城市颠猴、具體街道分行寫)
visa銀行卡信息:開戶行,開戶行所在地址小染,開戶行的郵政編碼翘瓮,開戶行持有人卡號(hào),開戶行持有人姓名
稅務(wù)信息:1.會(huì)問你是不是美國居民選擇NO. 2. 有沒有在美國從事商業(yè)性活動(dòng)裤翩,選擇NO. 之后填寫個(gè)人或組織名稱资盅,所在國家调榄,受益方式(獨(dú)立開發(fā)者選擇個(gè)人),居住地址呵扛,郵寄地址每庆,聲明人,頭銜
(2) 內(nèi)購產(chǎn)品id的配置 (開發(fā)人員配置)
如果是金幣或其它消耗品的產(chǎn)品的話用消耗性型項(xiàng)目今穿,參考名稱(內(nèi)購項(xiàng)目缤灵,比如金幣100),產(chǎn)品id,定價(jià)信息,使用內(nèi)購的快照蓝晒,顯示名稱腮出,描述。
(3) 用戶職能
測(cè)試員:添加水箱測(cè)試員及沙箱賬號(hào)芝薇,水箱測(cè)試賬號(hào)不能是正常使用的appid賬號(hào)胚嘲,直接使用一個(gè)沒有注冊(cè)過的郵箱賬號(hào)即可。
姓名洛二,測(cè)試賬號(hào)密碼馋劈,appstore地區(qū)(必須填對(duì))。
代碼如下 :
/*注意事項(xiàng):
1.沙盒環(huán)境測(cè)試appStore內(nèi)購流程的時(shí)候灭红,請(qǐng)使用沒越獄的設(shè)備侣滩。
2.請(qǐng)務(wù)必使用真機(jī)來測(cè)試,一切以真機(jī)為準(zhǔn)变擒。
3.項(xiàng)目的Bundle identifier需要與您申請(qǐng)AppID時(shí)填寫的bundleID一致君珠,不然會(huì)無法請(qǐng)求到商品信息。
4.如果是你自己的設(shè)備上已經(jīng)綁定了自己的AppleID賬號(hào)請(qǐng)先注銷掉,否則你哭爹喊娘都不知道是怎么回事娇斑。
5.訂單校驗(yàn) 蘋果審核app時(shí)策添,仍然在沙盒環(huán)境下測(cè)試,所以需要先進(jìn)行正式環(huán)境驗(yàn)證毫缆,如果發(fā)現(xiàn)是沙盒環(huán)境則轉(zhuǎn)到沙盒驗(yàn)證唯竹。
識(shí)別沙盒環(huán)境訂單方法:
1.根據(jù)字段 environment = sandbox。
2.根據(jù)驗(yàn)證接口返回的狀態(tài)碼,如果status=21007苦丁,則表示當(dāng)前為沙盒環(huán)境浸颓。
蘋果反饋的狀態(tài)碼:
21000App Store無法讀取你提供的JSON數(shù)據(jù)
21002 訂單數(shù)據(jù)不符合格式
21003 訂單無法被驗(yàn)證
21004 你提供的共享密鑰和賬戶的共享密鑰不一致
21005 訂單服務(wù)器當(dāng)前不可用
21006 訂單是有效的,但訂閱服務(wù)已經(jīng)過期旺拉。當(dāng)收到這個(gè)信息時(shí)产上,解碼后的收據(jù)信息也包含在返回內(nèi)容中
21007 訂單信息是測(cè)試用(sandbox),但卻被發(fā)送到產(chǎn)品環(huán)境中驗(yàn)證
21008 訂單信息是產(chǎn)品環(huán)境中使用蛾狗,但卻被發(fā)送到測(cè)試環(huán)境中驗(yàn)證
*/
#import <Foundation/Foundation.h>
typedef enum {
SIAPPurchSuccess = 0, // 購買成功
SIAPPurchFailed = 1, // 購買失敗
SIAPPurchCancle = 2, // 取消購買
SIAPPurchVerFailed = 3, // 訂單校驗(yàn)失敗
SIAPPurchVerSuccess = 4, // 訂單校驗(yàn)成功
SIAPPurchNotArrow = 5, // 不允許內(nèi)購
}SIAPPurchType;
typedef void (^IAPCompletionHandle)(SIAPPurchType type,NSData *data);
@interface STRIAPManager : NSObject
+ (instancetype)shareSIAPManager;
//開始內(nèi)購
- (void)startPurchWithID:(NSString *)purchID completeHandle:(IAPCompletionHandle)handle;
@end
.m
#import "STRIAPManager.h"
#import <StoreKit/StoreKit.h>
@interface STRIAPManager()<SKPaymentTransactionObserver,SKProductsRequestDelegate>{
NSString *_purchID;
IAPCompletionHandle _handle;
}
@end
@implementation STRIAPManager
#pragma mark - ??life cycle
+ (instancetype)shareSIAPManager{
static STRIAPManager *IAPManager = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken,^{
IAPManager = [[STRIAPManager alloc] init];
});
return IAPManager;
}
- (instancetype)init{
self = [super init];
if (self) {
// 購買監(jiān)聽寫在程序入口,程序掛起時(shí)移除監(jiān)聽,這樣如果有未完成的訂單將會(huì)自動(dòng)執(zhí)行并回調(diào) paymentQueue:updatedTransactions:方法
[[SKPaymentQueue defaultQueue] addTransactionObserver:self];
}
return self;
}
- (void)dealloc{
[[SKPaymentQueue defaultQueue] removeTransactionObserver:self];
}
#pragma mark - ??public
- (void)startPurchWithID:(NSString *)purchID completeHandle:(IAPCompletionHandle)handle{
if (purchID) {
if ([SKPaymentQueue canMakePayments]) {
// 開始購買服務(wù)
_purchID = purchID;
_handle = handle;
NSSet *nsset = [NSSet setWithArray:@[purchID]];
SKProductsRequest *request = [[SKProductsRequest alloc] initWithProductIdentifiers:nsset];
request.delegate = self;
[request start];
}else{
[self handleActionWithType:SIAPPurchNotArrow data:nil];
}
}
}
#pragma mark - ??private
- (void)handleActionWithType:(SIAPPurchType)type data:(NSData *)data{
#if DEBUG
switch (type) {
case SIAPPurchSuccess:
NSLog(@"購買成功");
break;
case SIAPPurchFailed:
NSLog(@"購買失敗");
break;
case SIAPPurchCancle:
NSLog(@"用戶取消購買");
break;
case SIAPPurchVerFailed:
NSLog(@"訂單校驗(yàn)失敗");
break;
case SIAPPurchVerSuccess:
NSLog(@"訂單校驗(yàn)成功");
break;
case SIAPPurchNotArrow:
NSLog(@"不允許程序內(nèi)付費(fèi)");
break;
default:
break;
}
#endif
if(_handle){
_handle(type,data);
}
}
#pragma mark - ??delegate
// 交易結(jié)束
- (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)證購買憑證
}
[self verifyPurchaseWithPaymentTransaction:transaction isTestServer:NO];
}
// 交易失敗
- (void)failedTransaction:(SKPaymentTransaction *)transaction{
if (transaction.error.code != SKErrorPaymentCancelled) {
[self handleActionWithType:SIAPPurchFailed data:nil];
}else{
[self handleActionWithType:SIAPPurchCancle data:nil];
}
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
}
- (void)verifyPurchaseWithPaymentTransaction:(SKPaymentTransaction *)transaction isTestServer:(BOOL)flag{
//交易驗(yàn)證
NSURL *recepitURL = [[NSBundle mainBundle] appStoreReceiptURL];
NSData *receipt = [NSData dataWithContentsOfURL:recepitURL];
if(!receipt){
// 交易憑證為空驗(yàn)證失敗
[self handleActionWithType:SIAPPurchVerFailed data:nil];
return;
}
// 購買成功將交易憑證發(fā)送給服務(wù)端進(jìn)行再次校驗(yàn)
[self handleActionWithType:SIAPPurchSuccess data:receipt];
NSError *error;
NSDictionary *requestContents = @{
@"receipt-data": [receipt base64EncodedStringWithOptions:0]
};
NSData *requestData = [NSJSONSerialization dataWithJSONObject:requestContents
options:0
error:&error];
if (!requestData) { // 交易憑證為空驗(yàn)證失敗
[self handleActionWithType:SIAPPurchVerFailed data:nil];
return;
}
//In the test environment, use https://sandbox.itunes.apple.com/verifyReceipt
//In the real environment, use https://buy.itunes.apple.com/verifyReceipt
NSString *serverString = @"https://buy.itunes.apple.com/verifyReceipt";
if (flag) {
serverString = @"https://sandbox.itunes.apple.com/verifyReceipt";
}
NSURL *storeURL = [NSURL URLWithString:serverString];
NSMutableURLRequest *storeRequest = [NSMutableURLRequest requestWithURL:storeURL];
[storeRequest setHTTPMethod:@"POST"];
[storeRequest setHTTPBody:requestData];
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
[NSURLConnection sendAsynchronousRequest:storeRequest queue:queue
completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
if (connectionError) {
// 無法連接服務(wù)器,購買校驗(yàn)失敗
[self handleActionWithType:SIAPPurchVerFailed data:nil];
} else {
NSError *error;
NSDictionary *jsonResponse = [NSJSONSerialization JSONObjectWithData:data options:0 error:&error];
if (!jsonResponse) {
// 蘋果服務(wù)器校驗(yàn)數(shù)據(jù)返回為空校驗(yàn)失敗
[self handleActionWithType:SIAPPurchVerFailed data:nil];
}
// 先驗(yàn)證正式服務(wù)器,如果正式服務(wù)器返回21007再去蘋果測(cè)試服務(wù)器驗(yàn)證,沙盒測(cè)試環(huán)境蘋果用的是測(cè)試服務(wù)器
NSString *status = [NSString stringWithFormat:@"%@",jsonResponse[@"status"]];
if (status && [status isEqualToString:@"21007"]) {
[self verifyPurchaseWithPaymentTransaction:transaction isTestServer:YES];
}else if(status && [status isEqualToString:@"0"]){
[self handleActionWithType:SIAPPurchVerSuccess data:nil];
}
#if DEBUG
NSLog(@"----驗(yàn)證結(jié)果 %@",jsonResponse);
#endif
}
}];
// 驗(yàn)證成功與否都注銷交易,否則會(huì)出現(xiàn)虛假憑證信息一直驗(yàn)證不通過,每次進(jìn)程序都得輸入蘋果賬號(hào)
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
}
#pragma mark - SKProductsRequestDelegate
- (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response{
NSArray *product = response.products;
if([product count] <= 0){
#if DEBUG
NSLog(@"--------------沒有商品------------------");
#endif
return;
}
SKProduct *p = nil;
for(SKProduct *pro in product){
if([pro.productIdentifier isEqualToString:_purchID]){
p = pro;
break;
}
}
#if DEBUG
NSLog(@"productID:%@", response.invalidProductIdentifiers);
NSLog(@"產(chǎn)品付費(fèi)數(shù)量:%lu",(unsigned long)[product count]);
NSLog(@"%@",[p description]);
NSLog(@"%@",[p localizedTitle]);
NSLog(@"%@",[p localizedDescription]);
NSLog(@"%@",[p price]);
NSLog(@"%@",[p productIdentifier]);
NSLog(@"發(fā)送購買請(qǐng)求");
#endif
SKPayment *payment = [SKPayment paymentWithProduct:p];
[[SKPaymentQueue defaultQueue] addPayment:payment];
}
//請(qǐng)求失敗
- (void)request:(SKRequest *)request didFailWithError:(NSError *)error{
#if DEBUG
NSLog(@"------------------錯(cuò)誤-----------------:%@", error);
#endif
}
- (void)requestDidFinish:(SKRequest *)request{
#if DEBUG
NSLog(@"------------反饋信息結(jié)束-----------------");
#endif
}
#pragma mark - SKPaymentTransactionObserver
- (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray<SKPaymentTransaction *> *)transactions{
for (SKPaymentTransaction *tran in transactions) {
switch (tran.transactionState) {
case SKPaymentTransactionStatePurchased:
[self completeTransaction:tran];
break;
case SKPaymentTransactionStatePurchasing:
#if DEBUG
NSLog(@"商品添加進(jìn)列表");
#endif
break;
case SKPaymentTransactionStateRestored:
#if DEBUG
NSLog(@"已經(jīng)購買過商品");
#endif
// 消耗型不支持恢復(fù)購買
[[SKPaymentQueue defaultQueue] finishTransaction:tran];
break;
case SKPaymentTransactionStateFailed:
[self failedTransaction:tran];
break;
default:
break;
}
}
}
@end
在控制器中調(diào)用晋涣,導(dǎo)入頭文件
調(diào)用方法
- (void)purchaseAction{
if (!_IAPManager) {
_IAPManager = [STRIAPManager shareSIAPManager];
}
// iTunesConnect 蘋果后臺(tái)配置的產(chǎn)品ID
[_IAPManager startPurchWithID:@"com.bb.helper_advisory" completeHandle:^(SIAPPurchType type,NSData *data) {
//請(qǐng)求事務(wù)回調(diào)類型,返回的數(shù)據(jù)
}];
}
經(jīng)典文章參考:
- [內(nèi)購流程] [ng]
[ng]: http://yimouleng.com/2015/12/17/ios-AppStore/ 內(nèi)購流程 - http://www.reibang.com/p/b199a4672608 完成交易后和服務(wù)器交互
- http://www.reibang.com/p/1ef61a785508 沙盒賬號(hào)測(cè)試