• iOS 开发 初级:应用内购买 In-App Purchase


    http://blog.csdn.net/songrotek/article/details/8680415

    现在有很多应用都使用了In-App Purchase,虽然对于很多用户来说,可能并不喜欢甚至讨厌这个模式,以为一点击就要从账户里扣钱。但是,应用内购买对于开发者而言不失为一种好的商业模式,而且人们也将越来越接受这种购买模式。

    下面开始介绍一下应用内购买的基本原理和编程方法。

    1、基本原理

    这里参考了Apple的开发文档In-App Purchase Programming Guide

    简要介绍一下整个流程:

    Pre 0:在iTunesConnect中对于的App创建相应的产品,并在应用信息中加入这些产品。具体步骤之后介绍。

    Step 1:应用内根据创建的产品的bundle identifier来获取产品的List。

    Step 2:应用请求产品的信息。产品信息为SKProduct对象。

    Step 3:App Store返回信息。在实际编程中,Step1,2,3是在一起的。通过创建SKProductsRequest得到SKProductsResponse。SKProducts信息就在SKProductsResponse的对象中,是其Property。

    Step 4:在应用中显示产品信息给用户

    Step 5:用户点击了一个产品。

    Step 6:应用向App Store发出一个购买请求 Payment Request

    Step 7:App Store处理请求,完成交易,并返回信息到应用。

    Step 8:  应用获取信息然后根据交易情况将购买的内容解锁给用户使用。

    这是In-App Purchase的一个基本的过程描述。在我们实际的编程过程中。对于这个产品列表,我们可能会选择直接提供给用户,而不是通过App Store获取信息。只有当用户点击了某个产品后,我们才开始去获取产品信息并完成购买。另一方面,在购买过程中,我们在应用中应该显示足够的提示信息,因此交易过程中的Notification也很重要。

    下面开始StepByStep介绍整个具体的实现过程。这里只是介绍最基本的实现方法,以non-consumable产品为例。

    Step 1:创建产品

    首先要说明一下为了实现应用内购买,你的AppID就是com.companyname.appname必须是唯一的,不能带*。

    在iTunesConnect中Manage My Applications中选择Manage In-App Purchases

    有四种产品类型,具体详见开发文档。这里选择non-consumable,就是一次购买终身受用的产品。consumable就是消费类可以不断购买的,这种在游戏中比较常见。

    上面是产品的详细信息填写。这里特别注意的是ProductID的填写,其实就是ProductIdentifier,这个和应用的BundleIdentifier类似,必须独一无二,一般的做法是填写成com.companyname.appname.productname,当然从本质上讲可以是任意字符串,只要独一无二就可以了。这个ProductID是之后在程序中获取产品信息的依据。其他方面的信息填写很简单,这里不在费述。

    Step 2:在应用版本信息中加入产品

    进入到应用页面,点击View Detail,然后在下面可以看到

    In-App Purchases,点击Edit然后加入之前创建的Products。

    Step 3:创建测试User

    为了在开发阶段测试In-App Purchase,Apple为我们提供了Test User功能,通过它,可以在开发时用这个账号免费实现应用内购买。

    具体就是在iTunesConnect首页,点击Manage Users

    点击Test User进行创建。

    Step 4:开始编程。在Xcode中要加入StoreKit.framework,通过它来实现功能

    Step 5:一般我们会单独创建一个类来实现应用内购买的功能。由于这个是教程,而不是案例,所以不打算把整个类的编写都搬进来。只是介绍一下重要的东西和流程。

    在类中要加入

    <SKProductsRequestDelegate,SKPaymentTransactionObserver>

    对于SKPaymentTransactionObserver这个东西可以监测交易的整个过程,即使交易时退出应用,交易也可以继续进行,当然要回到应用内的页面才能最后完成交易,显示产品相应内容。类的初始化应加入

    [[SKPaymentQueuedefaultQueue] addTransactionObserver:self];

    加入这句代码来实现TransactionObserver的功能,后面有相应的Methods可以加入

    paymentQueue:开头

    Step 6:下面的介绍不局限在一个类的编写,而是按照购买流程。假设我们已经编写好了一个应用内购买的类,然后我们要实现购买。首先就是Request Products。

    _productRequest = [[SKProductsRequestalloc]initWithProductIdentifiers:_productIdentifiers];

    _productRequest.delegate =self;

    [_productRequest start];


    一个完整的请求如上,对于productsIdentifiers,这是一个Set,就是在这里创建一个set加入各个ProductIdentifer

     NSSet *productIdentifiers = [NSSetsetWithObjects:

                                        K_CAMERA_ANGLE_MODE,

                                        K_SLOPE_ANGLE_MODE,

                                        K_DIHEDRAL_ANGLE_MODE,

                                        K_LINE_PLANE_ANGLE_MODE,

                                        nil];

    然后就是request delegate的methods

    - (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response

       NSArray *skProducts = response.products;

        // process....

    }

    - (void)request:(SKRequest *)request didFailWithError:(NSError *)error

    {

       // process....   

    }

    请求成功,就能获取Products,一个NSArray,里面就是SKProduct 对象的产品信息了。产品信息有名称,价格,等等,很容易找到。这些东西只是在显示信息时有用,购买时不需要,只要用SKProduct就行了

    Step 7:购买

    SKPayment *payment = [SKPaymentpaymentWithProduct:product];

    [[SKPaymentQueuedefaultQueue] addPayment:payment];

    代码如上。然后就开始连接App Store了。主要看下面

    #pragma mark - SKPaymentTransactionObserver

    - (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray*)transactions

    {

       for (SKPaymentTransaction *transactionin transactions) {

           switch (transaction.transactionState) {

                caseSKPaymentTransactionStatePurchased:

                    [selfcompleteTransaction:transaction];

                   break;

                caseSKPaymentTransactionStateFailed:

                    [selffailedTransaction:transaction];

                   break;

                caseSKPaymentTransactionStateRestored:

                    [selfrestoreTransaction:transaction];

                    

               default:

                   break;

            }

        }

    }


    这里说一下上面的第三种交易状态Restore,恢复。这个是这样的。如果有些人在iPhone上用一个账号购买了一个产品,那么在iPad上又下载了这个应用,还要再重新购买吗?不用了,通过Restore在App Store中检测你这个账号的购买记录,如果有购买记录存在,那就不用再次购买了,直接restoreTransaction。

    接下来就是根据购买的状态分别进行处理

    - (void) completeTransaction: (SKPaymentTransaction *)transaction
    {
    
        // Your application should implement these two methods.
        [self recordTransaction:transaction];
        [self provideContent:transaction.payment.productIdentifier];
    
        // Remove the transaction from the payment queue.
    
        [[SKPaymentQueue defaultQueue] finishTransaction: transaction];
    }
    

    一般我们用NSUserDefaults来进行交易的记录就行了。

    注意程序中finishTransaction这一行代码,这样TransactionObserver就不再监测这个交易了。

    其他状态Restore,failed都是差不多的处理,这些代码在开发文档中有。

    当然,对于restore,还有一个Method要注意

    - (void)paymentQueue:(SKPaymentQueue *)queue restoreCompletedTransactionsFailedWithError:(NSError *)error

    如果restore失败,可以显示相应的信息提示

    Step 8:附加

    在整个购买过程中,我们一般要给用户一下提示信息,比如等待,比如正在连接,比如交易已完成。要实现这些功能,就应该用notification,在上面的交易环节加入postNotification,然后对notification进行有效处理。本文只讲In-App Purchase,关于notification的编程不做介绍。

    基本上,通过上面的环节就能完成整个应用内购买了。当然,从安全性上考虑,Apple在交易完成后会发送验证信息,通过发送验证信息给App Store来判断这笔交易是否出自App Store,从而确认交易的合法性。

    关于这方面,Apple 有一个代码包提供了validationController来实现很方便的验证。

  • 相关阅读:
    【代码片段】HTML5嵌入媒体 (HTML5 Embedded Media)
    【代码片端】@Font-Face
    【代码片段】CSS3 渐变(CSS3 Gradients)
    【代码片段】CSS RESET,支持HTML5
    我的Cocos2d-x学习笔记(九)游戏帧循环(游戏主循环)
    我的Cocos2d-x学习笔记(八)利用CCSpriteBatchNode进行优化
    我的Cocos2d-x学习笔记(七)纹理缓存、帧缓存、精灵的创建、zOrder
    我的Cocos2d-x学习笔记(六)坐标体系与锚点
    我的Cocos2d-x学习笔记(五)CCLayer、CCLayerColor、CCLayerGradient、CCLayerMultiplex
    我的Cocos2d-x学习笔记(四)场景转换特效
  • 原文地址:https://www.cnblogs.com/123ing/p/4097663.html
Copyright © 2020-2023  润新知