*步:创建一个Plist文件,保存自己的商品id(prodectId)

%title插图%num

第二步:获取商品ID,向苹果服务器请求自己的可买的商品(products),结果通过代理方法返回
NSString *path = [[NSBundlemainBundle] pathForResource:@”iap.plist”ofType:nil];
NSArray *products = [NSArrayarrayWithContentsOfFile:path];

//获取所有的productid
NSArray *productidA = [productsvalueForKeyPath:@”productid”];

//获取productid的set(集合)
NSSet *set = [NSSetsetWithArray:productidA];

//向苹果发送请求,请求可买商品
self.productsRequest = [[SKProductsRequestalloc]initWithProductIdentifiers:set];
self.productsRequest.delegate =self;
[self.productsRequeststart];

第三步:苹果服务器返回商品数据会调用代理方法,可以将商品进行排序

#pragma mark – SKProductsRequestDelegate
/**
* 当请求到可卖商品的结果会执行该方法
*
* @param response response中存储了可卖商品的结果
*/
// Sent immediately before -requestDidFinish:
-(void)productsRequest:(SKProductsRequest *)request didReceiveResponse: (SKProductsResponse *)response {
self.products = response.products;
//将商品经行排序
self.products = [self.productssortedArrayWithOptions:NSSortConcurrent usingComparator:^NSComparisonResult(SKProduct *obj1,SKProduct *obj2) {
return [obj1.pricecompare:obj2.price];
}];
}

第四步:添加实时观察者,用户购买行为通过观察者模式进行回调,在viewWillAppear:方法里面添加观察者,后面在viewDidDisappear:方法里面移除观察者,防止其他页面使用内购,影响此页面的逻辑
– (void)viewWillAppear:(BOOL)animated {
[superviewWillAppear:animated];
// 添加观察者监听交易状态
[[SKPaymentQueuedefaultQueue] addTransactionObserver:self];
}

第五步:根据苹果服务器返回的商品,选择商品ID发送购买请求(注意:可能苹果测试服务器慢,主要做一个商品ID容错判断,防止崩溃),结果通过观察者模式回调回来,结果返回时,要将交易从队列中移除
/* 1.取出模型*/
SKProduct *product = [self.productslastObject];
if (product.productIdentifier) {
// 2.购买商品
[self buyProduct:product];
} else {
[MBProgressHUDshowError:@”系统繁忙,请您稍后再试!”];
}

/**
* 购买商品
*
* @param product 商品数据
*/
– (void)buyProduct:(SKProduct *)product {

[MBProgressHUD showMessage:@”购买中…” toView:self.view];
//创建票据
SKPayment *payment = [SKPayment paymentWithProduct:product];

//将票据加入到交易队列中
[[SKPaymentQueue defaultQueue] addPayment:payment];

}

/**
* 当交易队列中的交易状态发生改变的时候会执行该方法
*
* @param transactions 数组中存放了所有的交易
*/
– (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transactions
{
/*
SKPaymentTransactionStatePurchasing, 正在购买
SKPaymentTransactionStatePurchased, 购买完成(销毁交易)
SKPaymentTransactionStateFailed, 购买失败(销毁交易)
SKPaymentTransactionStateRestored, 恢复购买(销毁交易)
SKPaymentTransactionStateDeferred *终状态未确定
*/
for (SKPaymentTransaction *transactionin transactions) {
switch (transaction.transactionState) {
caseSKPaymentTransactionStatePurchasing:
NSLog(@”用户正在购买”);
break;

caseSKPaymentTransactionStatePurchased:
NSLog(@”购买成功”);
[queue finishTransaction:transaction];
break;

caseSKPaymentTransactionStateFailed:
[queue finishTransaction:transaction];
break;

caseSKPaymentTransactionStateRestored:
NSLog(@”恢复购买”);
[queue finishTransaction:transaction];
break;

caseSKPaymentTransactionStateDeferred:
NSLog(@”*终状态未确定”);
break;

default:
break;
}
}
}

第六步:根据苹果服务器返回的购买结果,进行相应的处理,购买成功后可以讲票据发送给后台,后台进行二次验证,保存票据信息或者是本地进行验证

//1.服务器验证
/**
* 发送用户数据给服务器
*/
– (void)sendUserPayMessageToSever {

__weak __typeof(&*self)weakSelf = self;
// 验证凭据,获取到苹果返回的交易凭据
// appStoreReceiptURL iOS7.0增加的,购买交易完成后,会将凭据存放在该地址
NSURL *receiptURL = [[NSBundlemainBundle] appStoreReceiptURL];
// 从沙盒中获取到购买凭据
NSData *receiptData = [NSDatadataWithContentsOfURL:receiptURL];
/**
BASE64 常用的编码方案,通常用于数据传输,以及加密算法的基础算法,传输过程中能够保证数据传输的稳定性
BASE64是可以编码和解码的
*/
NSString *encodeStr = [receiptData base64EncodedStringWithOptions:NSDataBase64EncodingEndLineWithLineFeed];

NSDictionary *parameters =@{@”receipt”:encodeStr};

[XYNetworkServermethodPostWithUrl:IPAUrlparameters:parameters SuccessBlock:^(id responseObject) {
NSLog(@”%@”,responseObject);
} andErrorBlock:^(id error) {
NSLog(@”%@”,error);
}];

}

//2.本地验证
– (void)verifyPruchase{
// 验证凭据,获取到苹果返回的交易凭据
// appStoreReceiptURL iOS7.0增加的,购买交易完成后,会将凭据存放在该地址
NSURL *receiptURL = [[NSBundlemainBundle] appStoreReceiptURL];
// 从沙盒中获取到购买凭据
NSData *receiptData = [NSDatadataWithContentsOfURL:receiptURL];
// 发送网络POST请求,对购买凭据进行验证
NSURL *url = [NSURLURLWithString:Url];
// 国内访问苹果服务器比较慢,timeoutInterval需要长一点
NSMutableURLRequest *request = [NSMutableURLRequestrequestWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicytimeoutInterval:10.0f];

request.HTTPMethod =@”POST”;

// 在网络中传输数据,大多情况下是传输的字符串而不是二进制数据
// 传输的是BASE64编码的字符串
/**
BASE64 常用的编码方案,通常用于数据传输,以及加密算法的基础算法,传输过程中能够保证数据传输的稳定性
BASE64是可以编码和解码的
*/
NSString *encodeStr = [receiptData base64EncodedStringWithOptions:NSDataBase64EncodingEndLineWithLineFeed];

NSString *payload = [NSStringstringWithFormat:@”{\”receipt-data\” : \”%@\”}”, encodeStr];
NSData *payloadData = [payloaddataUsingEncoding:NSUTF8StringEncoding];

request.HTTPBody = payloadData;

// 提交验证请求,并获得官方的验证JSON结果
NSData *result = [NSURLConnectionsendSynchronousRequest:requestreturningResponse:nilerror:nil];

// 官方验证结果为空
if (result ==nil) {
NSLog(@”验证失败”);
}

NSDictionary *dict = [NSJSONSerializationJSONObjectWithData:resultoptions:NSJSONReadingAllowFragmentserror:nil];

NSLog(@”%@”, dict);

if (dict !=nil) {
// 比对字典中以下信息基本上可以保证数据安全
// bundle_id&application_version&product_id&transaction_id
NSLog(@”验证成功”);
}
}

第七步:移除观察者(注意:因为苹果服务器比较慢,这里要将SKProductsRequest代理对象的代理置空,防止苹果服务器返回数据时,控制器销毁了,找不到代理对象,奔溃报野指针错误)

– (void)viewDidDisappear:(BOOL)animated {
[superviewDidDisappear:animated];

#warning 移除向苹果服务器请求(SKProductsRequest)的地理对象,否则可能会奔溃
if ([selfisMovingFromParentViewController]) {
if (self.productsRequest.delegate == self) {
self.productsRequest.delegate =nil;
}
}
// 移除观察交易状态的观察者
[[SKPaymentQueuedefaultQueue] removeTransactionObserver:self];
}

第八步:为了更加安全的运行程序,在dealloc移除SKProductsRequest代理对象的代理置空,防止苹果服务器返回数据时,控制器销毁了,找不到代理对象,奔溃报野指针错误

– (void)dealloc {
#warning 移除向苹果服务器请求(SKProductsRequest)的地理对象,否则可能会奔溃
if (self.productsRequest.delegate == self) {
self.productsRequest =nil;
}
[[NSNotificationCenterdefaultCenter] removeObserver:self];
NSLog(@”——- dealloc”);
}