iOS之支付

iOS支付

iOS支付分为两类,第三方支付应用内支付(内购)

第三方支付包括:支付宝支付、微信支付、银联支付、百度钱包、京东支付等等。

应用内支付(In-App Purchase):在应用程序内购买虚拟商品。如果你在App Store上销售的应用程序,将收到支付金额的70%。

第三方支付

弹出方式

网页

有些第三方支付没有安装客户端,可以直接弹出网页进行支付。(比如支付宝)

调用APP

手机中安装了客户端可以跳转到APP中进行支付。微信支付只能调用App进行支付。

支付宝支付

相关资料

支付流程

  1. 在商户服务平台先与支付宝签约,获得商户ID(partner)和账号ID(seller),需要提供公司资质或者营业执照,个人无法申请。

    文档地址:
    https://doc.open.alipay.com/doc2/detail?treeId=58&articleId=103542&docType=1

  2. 生成并下载相应的公钥私钥文件(加密签名用)

    文档地址:
    https://doc.open.alipay.com/doc2/detail.htm?spm=0.0.0.0.POMYKl&treeId=58&articleId=103543&docType=1

  3. 下载支付宝SDK:
    https://doc.open.alipay.com/doc2/detail?0treeId=54&articleId=103419&docType=1

  4. 生成订单信息

  5. 调用支付宝客户端,由支付宝客户端跟支付宝安全服务器打交道

  6. 支付完毕后返回支付结果给商户客户端和服务器

SDK里有集成支付宝功能的一个Demo,集成支付功能的具体操作方式,可以参考Demo。

代码集成流程

参考文档地址:
https://doc.open.alipay.com/doc2/detail.htm?spm=0.0.0.0.efmKDS&treeId=59&articleId=103676&docType=1

  1. 下载官方SDK

    下载地址:
    https://doc.open.alipay.com/doc2/detail?treeId=54&articleId=103419&docType=1

    本Demo使用的SDK是从官方Demo整理出来的,整理的SDK版本:201501022。

    下载地址:http://7xooko.com1.z0.glb.clouddn.com/AlipaySDK.zip

    目录结构如下:

    ├── AlipaySDK.bundle
    ├── AlipaySDK.framework
    ├── Order.h
    ├── Order.m
    ├── Util
    ├── libcrypto.a
    ├── libssl.a
    └── openssl
    

    其中:

    • AlipaySDK.bundleAlipaySDK.framework是支付宝SDK
    • Order类:定义订单信息
    • Util、libcrypto.a、libssl.a、openssl:数据签名,对订单信息进行加密
  2. 添加依赖库

    其中,需要注意的是:

    如果是Xcode 7.0之后的版本,需要添加libc++.tbd、libz.tbd;

    如果是Xcode 7.0之前的版本,需要添加libc++.dylib、libz.dylib。

  3. 创建prefix header filePCH文件,添加#import <Foundation/Foundation.h>

    Build Settings中的prefix header设置pch文件路径

  4. Build SettingsHeader Search Paths添加头文件引用路径,[文件路径]/AlipaySDK/

  5. 在需要调用AlipaySDK的文件中,增加头文件引用。

    #import  <AlipaySDK/AlipaySDK.h>
    #import "Order.h"
    #import "DataSigner.h"
    
  6. 生成订单信息及签名

    //将商品信息赋予AlixPayOrder的成员变量
    Order *order = [[Order alloc] init];
    order.partner = PartnerID; // 商户ID
    order.seller = SellerID; // 账号ID
    order.tradeNO = @"20150923"; //订单ID(由商家自行制定)
    order.productName = @"iPhone6s"; //商品标题
    order.productDescription = @"新年打折"; //商品描述
    order.amount = @"0.01"; //商品价格(单位:元)
    order.notifyURL =  @"http://www.chaosky.me"; //回调URL,支付成功或者失败回调通知自己的服务器进行订单状态变更
    order.service = @"mobile.securitypay.pay";
    order.paymentType = @"1";
    order.inputCharset = @"utf-8";
    order.itBPay = @"30m";
    order.showUrl = @"m.alipay.com";
    
    // 应用注册scheme,在AlixPayDemo-Info.plist定义URL types
    NSString *appScheme = @"AliPayDemo";
    
    //将商品信息拼接成字符串
    NSString *orderSpec = [order description];
    NSLog(@"orderSpec = %@",orderSpec);
    
    //获取私钥并将商户信息签名,外部商户可以根据情况存放私钥和签名,只需要遵循RSA签名规范,并将签名字符串base64编码和UrlEncode
    id<DataSigner> signer = CreateRSADataSigner(PartnerPrivKey);
    NSString *signedString = [signer signString:orderSpec];
    
    //将签名成功字符串格式化为订单字符串,请严格按照该格式
    NSString *orderString = nil;
    if (signedString != nil) {
        orderString = [NSString stringWithFormat:@"%@&sign=\"%@\"&sign_type=\"%@\"",
                       orderSpec, signedString, @"RSA"];
    
        [[AlipaySDK defaultService] payOrder:orderString fromScheme:appScheme callback:^(NSDictionary * resultDic) {
            NSLog(@"reslut = %@",resultDic);
        }];
    }
    
  7. Xcode设置URL scheme

    iPhone SDK可以把你的App和一个自定义的URL Scheme绑定。该URL Scheme可用来从浏览器或别的App启动你的App。

    配置方法:打开info.plist文件,找到或者添加如图所示的键值对:

    URL Scheme值为代码中对应的值,必须一致

  8. 配置支付宝客户端返回url处理方法

    AppDelegate.m文件中,增加引用代码:

    #import <AlipaySDK/AlipaySDK.h>
    

    在@implementation AppDelegate中增加如下代码:

    - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation
    {
        //如果极简开发包不可用,会跳转支付宝钱包进行支付,需要将支付宝钱包的支付结果回传给开发包
        if ([url.host isEqualToString:@"safepay"]) {
            [[AlipaySDK defaultService] processOrderWithPaymentResult:url standbyCallback:^(NSDictionary * resultDic) {
        //【由于在跳转支付宝客户端支付的过程中,商户app在后台很可能被系统kill了,所以pay接口的callback就会失效,请商户对standbyCallback返回的回调结果进行处理,就是在这个方法里面处理跟callback一样的逻辑】
                NSLog(@"result = %@",resultDic);
            }];
        }
        if ([url.host isEqualToString:@"platformapi"]){//支付宝钱包快登授权返回authCode
    
            [[AlipaySDK defaultService] processAuthResult:url standbyCallback:^(NSDictionary * resultDic) {
                //【由于在跳转支付宝客户端支付的过程中,商户app在后台很可能被系统kill了,所以pay接口的callback就会失效,请商户对standbyCallback返回的回调结果进行处理,就是在这个方法里面处理跟callback一样的逻辑】
                NSLog(@"result = %@",resultDic);
            }];
        }
        return YES;
    }
    

微信支付

需要提供公司资质或者营业执照,个人无法申请。

相关文档

支付流程

  1. 向微信注册你的应用程序id

    开发者应用登记页面 进行登记,登记并选择移动应用进行设置后,将获得AppID,可立即用于开发。但应用登记完成后还需要提交审核,只有审核通过的应用才能正式发布使用。

  2. 微信APP支付接入商户服务中心

    参考文档链接:https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=open1419317780&token=&lang=zh_CN

  3. 下载微信SDK文件,如果在项目中应使用SDK的最新版。

    官方资源下载地址:https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&verify=1&id=open1419319164&token=&lang=zh_CN

    本Demo使用的SDK是从官方Demo整理出来的,整理的SDK版本:1.6.1。

    下载地址:http://7xooko.com1.z0.glb.clouddn.com/AlipaySDK.zip

    目录结构如下:

    ├── SDKExport
    │   ├── WXApi.h
    │   ├── WXApiObject.h
    │   ├── libWeChatSDK.a
    │   └── read_me.txt
    └── lib
        ├── ApiXml.h
        ├── ApiXml.mm
        ├── WXUtil.h
        ├── WXUtil.mm
        ├── payRequsestHandler.h
        └── payRequsestHandler.mm
    

    其中:

    SDKExport文件夹:SDK文件

    lib文件夹:工具类

  4. 添加依赖库

    SystemConfiguration.framework
    libz.dylib
    libsqlite3.dylib
    libc++.dylib
    CoreTelephony.framework
    CoreGraphics.framework
    
  5. 在需要调用WeChatSDK的文件中,增加头文件引用。

    #import "WXApi.h"
    #import "payRequsestHandler.h"
    
  6. 生成订单信息及签名

    #pragma mark - 微信支付
    - (void)wechatPayAction:(UIButton *) sender
    {
        // 判断用户是否安装微信
        if (![WXApi isWXAppInstalled]) {
            UIAlertView * alertView = [[UIAlertView alloc] initWithTitle:@"提示" message:@"请安装微信客户端" delegate:nil cancelButtonTitle:@"确定" otherButtonTitles:nil];
            [alertView show];
            return;
        }
        
        // 实现支付
        [self sendPay_demo];
    }
    
    - (void)sendPay_demo
    {
        //{{{
        //本实例只是演示签名过程, 请将该过程在商户服务器上实现
        
        // 配置微信支付的参数
        //创建支付签名对象
        payRequsestHandler *req = [[payRequsestHandler alloc] init];
        //初始化支付签名对象
        [req init:__WXappID mch_id:__WXmchID];
        //设置密钥
        [req setKey:__WXpaySignKey];
        
        //}}}
        
        //获取到实际调起微信支付的参数后,在app端调起支付
        NSMutableDictionary *dict = [req sendPay_demo];
        
        if(dict == nil){
            //错误提示
            NSString *debug = [req getDebugifo];
            
            [self alert:@"提示信息" msg:debug];
            
            NSLog(@"%@\n\n",debug);
        }else{
            NSLog(@"%@\n\n",[req getDebugifo]);
            //[self alert:@"确认" msg:@"下单成功,点击OK后调起支付!"];
            
            NSMutableString *stamp  = [dict objectForKey:@"timestamp"];
            
            //调起微信支付
            PayReq* req             = [[PayReq alloc] init];
            req.openID              = [dict objectForKey:@"appid"];
            req.partnerId           = [dict objectForKey:@"partnerid"];
            req.prepayId            = [dict objectForKey:@"prepayid"];
            req.nonceStr            = [dict objectForKey:@"noncestr"];
            req.timeStamp           = stamp.intValue;
            req.package             = [dict objectForKey:@"package"];
            req.sign                = [dict objectForKey:@"sign"];
            
            // 调用微信支付
            [WXApi sendReq:req];
        }
    }
    
    //客户端提示信息
    - (void)alert:(NSString *)title msg:(NSString *)msg
    {
        UIAlertView *alter = [[UIAlertView alloc] initWithTitle:title message:msg delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil];
        
        [alter show];
    }
    

  7. Xcode设置URL scheme

    在Xcode中,选择你的工程设置项,选中“TARGETS”一栏,在“info”标签栏的“URL type“添加“URL scheme”为你所注册的应用程序id(如下图所示)。

  8. 在你需要使用微信终端API的文件中import WXApi.h 头文件,并增加 WXApiDelegate 协议。

    // 微信所有的API接口
    #import "WXApi.h"
    // APP端签名相关头文件
    #import "payRequsestHandler.h"
    @interface AppDelegate ()<WXApiDelegate>
    @end
    
  9. 要使你的程序启动后微信终端能响应你的程序,必须在代码中向微信终端注册你的id。(如下图所示,在 AppDelegate 的 didFinishLaunchingWithOptions 函数中向微信注册id)。

    - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
        // Override point for customization after application launch.
        //向微信注册
        [WXApi registerApp:APP_ID withDescription:@"demo 2.0"];
        return YES;
    }
    

    重写AppDelegate的handleOpenURL和openURL方法:

    - (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url
    {
        return [WXApi handleOpenURL:url delegate:self];
    }
    
    - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation
    {
        return [WXApi handleOpenURL:url delegate:self];
    }
    
  10. 现在,你的程序要实现和微信终端交互的具体请求与回应,因此需要实现WXApiDelegate协议的两个方法:

    -(void) onReq:(BaseReq*)req
    {
        if([req isKindOfClass:[GetMessageFromWXReq class]])
        {
            // 微信请求App提供内容, 需要app提供内容后使用sendRsp返回
            NSString * strTitle = [NSString stringWithFormat:@"微信请求App提供内容"];
            NSString * strMsg = @"微信请求App提供内容,App要调用sendResp:GetMessageFromWXResp返回给微信";
    
            UIAlertView * alert = [[UIAlertView alloc] initWithTitle:strTitle message:strMsg delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
            alert.tag = 1000;
            [alert show];
        }
        else if([req isKindOfClass:[ShowMessageFromWXReq class]])
        {
            ShowMessageFromWXReq *  temp = (ShowMessageFromWXReq*)req;
            WXMediaMessage * msg = temp.message;
    
            //显示微信传过来的内容
            WXAppExtendObject * obj = msg.mediaObject;
    
            NSString * strTitle = [NSString stringWithFormat:@"微信请求App显示内容"];
            NSString * strMsg = [NSString stringWithFormat:@"标题:%@ \n内容:%@ \n附带信息:%@ \n缩略图:%lu bytes\n\n", msg.title, msg.description, obj.extInfo, msg.thumbData.length];
    
            UIAlertView * alert = [[UIAlertView alloc] initWithTitle:strTitle message:strMsg delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
            [alert show];
        }
        else if([req isKindOfClass:[LaunchFromWXReq class]])
        {
            //从微信启动App
            NSString * strTitle = [NSString stringWithFormat:@"从微信启动"];
            NSString * strMsg = @"这是从微信启动的消息";
            UIAlertView * alert = [[UIAlertView alloc] initWithTitle:strTitle message:strMsg delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
            [alert show];
        }
    }
    

    onReq是微信终端向第三方程序发起请求,要求第三方程序响应。第三方程序响应完后必须调用sendRsp返回。在调用sendRsp返回时,会切回到微信终端程序界面。

    -(void) onResp:(BaseResp*)resp
     {
         NSString * strMsg = [NSString stringWithFormat:@"errcode:%d", resp.errCode];
         NSString * strTitle;
    
         if([resp isKindOfClass:[SendMessageToWXResp class]])
         {
             strTitle = [NSString stringWithFormat:@"发送媒体消息结果"];
         }
         if([resp isKindOfClass:[PayResp class]]){
             //支付返回结果,实际支付结果需要去微信服务器端查询
             strTitle = [NSString stringWithFormat:@"支付结果"];
    
             switch (resp.errCode) {
                 case WXSuccess:
                     strMsg = @"支付结果:成功!";
                     NSLog(@"支付成功-PaySuccess,retcode = %d", resp.errCode);
                     break;
    
                 default:
                     strMsg = [NSString stringWithFormat:@"支付结果:失败!retcode = %d, retstr = %@", resp.errCode,resp.errStr];
                     NSLog(@"错误,retcode = %d, retstr = %@", resp.errCode,resp.errStr);
                     break;
             }
         }
         UIAlertView * alert = [[UIAlertView alloc] initWithTitle:strTitle message:strMsg delegate:self cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
         [alert show];
     }
    

    如果第三方程序向微信发送了sendReq的请求,那么onResp会被回调。sendReq请求调用后,会切到微信终端程序界面。

应用内购买(In-App Purchase)

在应用程序内购买虚拟商品。如果你在App Store上销售的应用程序,将收到支付金额的70%。

相关资料

支付流程

配置App ID

  1. 为应用建立建立一个不带通配符的App ID
  2. 用该App ID生成和安装相应的Provisioning Profile文件。

配置iTunes Connect

  1. 填写相关的税务,银行,联系人信息

    参考链接:iOS App提交指南(二)-协议、税务和银行业务

  2. 添加一个用于在sandbox付费的测试用户

  3. 用该App ID创建一个新的应用。

  4. 创建应用内付费项目,选择付费类型。

    App 内购买项目摘要填写

主要代码实现

  1. 在工程中引入 StoreKit.framework#import <StoreKit/StoreKit.h>

  2. 获得所有的付费Product ID列表。这个可以用常量存储在本地,也可以由自己的服务器返回。

    //在内购项目中创建的商品单号
    #define ProductID_IAP_FTHJ @"com.1000phone.IAPDemo.fthj_purple" // 方天画戟 488元
    #define ProductID_IAP_XYJ @"com.1000phone.IAPDemo.xyj" // 轩辕剑 6,498元
    #define ProductID_IAP_JB @"com.1000phone.IAPDemo.jb" // 金币 6元=6金币
    

  3. 制作界面,展示所有的应用内付费项目。这些应用内付费项目的价格和介绍信息可以从App Store服务器请求,也可以是自己的服务器返回。向App Store查询速度非常慢,通常需要2-3秒钟,最好从服务器请求。

    - (void)createViews
    {
        NSArray * buttonNames = @[@"轩辕剑 6498元", @"方天画戟 488元", @"金币6元=6金币"];
        __weak typeof(self) weakSelf = self;
        [buttonNames enumerateObjectsUsingBlock:^(NSString * buttonName, NSUInteger idx, BOOL * stop) {
            UIButton * button = [UIButton buttonWithType:UIButtonTypeSystem];
            [weakSelf.view addSubview:button];
            button.frame = CGRectMake(100, 100 + idx   * 60, 150, 50);
            button.titleLabel.font = [UIFont systemFontOfSize:18];
            [button setTitle:buttonName forState:UIControlStateNormal];
    
            // 设置tag值
            button.tag = PAY_BUTTON_BEGIN_TAG + idx;
            [button addTarget:self action:@selector(buyProduct:) forControlEvents:UIControlEventTouchUpInside];
        }];
    }
    
    - (void)buyProduct:(UIButton *) sender
    {
    
    }
    

  4. 当用户点击了一个IAP项目,我们先查询用户是否允许应用内付费。

    - (void)buyProduct:(UIButton *) sender
    {
        self.buyType = sender.tag - PAY_BUTTON_BEGIN_TAG;
        if ([SKPaymentQueue canMakePayments]) {
            // 执行下面提到的第5步:
            [self requestProductData];
            NSLog(@"允许程序内付费购买");
        }
        else
        {
            NSLog(@"不允许程序内付费购买");
            UIAlertView *alerView =  [[UIAlertView alloc] initWithTitle:@"提示"
                                                                message:@"您的手机没有打开程序内付费购买"
                                                               delegate:nil cancelButtonTitle:NSLocalizedString(@"关闭",nil) otherButtonTitles:nil];
    
            [alerView show];
    
        }
    }
    
  5. 我们先通过该IAP的ProductID向AppStore查询,获得SKPayment实例,然后通过SKPaymentQueue的 addPayment方法发起一个购买的操作。

    // 下面的ProductId应该是事先在itunesConnect中添加好的,已存在的付费项目。否则查询会失败。
    - (void)requestProductData {
       NSLog(@"---------请求对应的产品信息------------");
       NSArray *product = nil;
       switch (self.buyType) {
           case 0:
               product = [NSArray arrayWithObject:ProductID_IAP_XYJ];
               break;
           case 1:
               product = [NSArray arrayWithObject:ProductID_IAP_FTHJ];
               break;
           case 2:
               product = [NSArray arrayWithObject:ProductID_IAP_JB];
               break;
       }
       NSSet *nsset = [NSSet setWithArray:product];
       SKProductsRequest *request=[[SKProductsRequest alloc] initWithProductIdentifiers: nsset];
       request.delegate=self;
       [request start];
    }
    
    #pragma mark - SKProductsRequestDelegate
    // 收到的产品信息回调
    - (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response{
    
       NSLog(@"-----------收到产品反馈信息--------------");
       NSArray *myProduct = response.products;
       if (myProduct.count == 0) {
           NSLog(@"无法获取产品信息,购买失败。");
           return;
       }
       NSLog(@"产品Product ID:%@",response.invalidProductIdentifiers);
       NSLog(@"产品付费数量: %d", (int)[myProduct count]);
       // populate UI
       for(SKProduct *product in myProduct){
           NSLog(@"product info");
           NSLog(@"SKProduct 描述信息%@", [product description]);
           NSLog(@"产品标题 %@" , product.localizedTitle);
           NSLog(@"产品描述信息: %@" , product.localizedDescription);
           NSLog(@"价格: %@" , product.price);
           NSLog(@"Product id: %@" , product.productIdentifier);
       }
       SKPayment * payment = [SKPayment paymentWithProduct:myProduct[0]];
       NSLog(@"---------发送购买请求------------");
       [[SKPaymentQueue defaultQueue] addPayment:payment];
    
    }
    
    //弹出错误信息
    - (void)request:(SKRequest *)request didFailWithError:(NSError *)error{
       NSLog(@"-------弹出错误信息----------");
       UIAlertView *alerView =  [[UIAlertView alloc] initWithTitle:NSLocalizedString(@"Alert",NULL) message:[error localizedDescription]
                                                          delegate:nil cancelButtonTitle:NSLocalizedString(@"Close",nil) otherButtonTitles:nil];
       [alerView show];
    
    }
    
    -(void) requestDidFinish:(SKRequest *)request
    {
       NSLog(@"----------反馈信息结束--------------");
    
    }
    
  6. 在viewDidLoad方法中,将购买页面设置成购买的Observer。

    - (void)viewDidLoad {
        [super viewDidLoad];
        [self createViews];
        // 监听购买结果
        [[SKPaymentQueue defaultQueue] addTransactionObserver:self];
    }
    
    - (void)dealloc
    {
        [[SKPaymentQueue defaultQueue] removeTransactionObserver:self];
    }
    
  7. 当用户购买的操作有结果时,就会触发下面的回调函数,相应进行处理即可。

    #pragma mark - SKPaymentTransactionObserver
    // 处理交易结果
    - (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://已经购买过该商品
                    [self restoreTransaction:transaction];
                    break;
                case SKPaymentTransactionStatePurchasing:      //商品添加进列表
                    NSLog(@"商品添加进列表");
                    break;
                default:
                    break;
            }
        }
    
    }
    
    // 交易完成
    - (void)completeTransaction:(SKPaymentTransaction *)transaction {
        NSString * productIdentifier = transaction.payment.productIdentifier;
    //    NSString * receipt = [transaction.transactionReceipt base64EncodedString];
        if ([productIdentifier length] > 0) {
            // 向自己的服务器验证购买凭证
        }
    
        // 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 {
        // 对于已购商品,处理恢复购买的逻辑
        [[SKPaymentQueue defaultQueue] finishTransaction: transaction];
    }
    
  8. 服务器验证凭证(Optional)。如果购买成功,我们需要将凭证发送到服务器上进行验证。考虑到网络异常情况,iOS端的发送凭证操作应该进行持久化,如果程序退出,崩溃或网络异常,可以恢复重试。

参考链接

  1. iOS开发内购全套图文教程
  2. iOS应用内付费(IAP)开发步骤列表
  3. iOS内购实现及测试Check List

苹果支付( Pay)

苹果支付是一种在应用内运行的具有隐秘性和安全性非接触式的支付方式。它允许触摸付款,你可以用来购买实体商品和服务。

Apple 不会存储或共享客户的实际信用卡和借记卡卡号,因此商家和 App 开发者无需负责管理和保护实际的信用卡和借记卡卡号。

先决条件

除了使用 PassKit 框架实施 Apple Pay 之外,您还必须:

参考资料

  1. 官方Pay教程
  2. Apple Pay 中文入门

 Pay VS In-App Purchase

 Pay In-App Purchase
框架 PassKit StoreKit
适用范围 实体商品(如食品杂货、服装和电器)和服务(如俱乐部会员、酒店预订和活动门票) 销售虚拟商品,如适用于您的 App 的优质内容及订阅数字内容;程序内的内容和功能性;程序内货币服务;数码订阅
支付处理 自己的支付平台处理付款 苹果公司处理付款
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • 在我们应用开发中我们经常在自己的项目中使用到支付,下面我们来谈谈iOS这块的支付;iOS支付主要分为两类,第三方支...
    Hither阅读 8,254评论 9 42
  • iOS支付 iOS支付分为两类,第三方支付和应用内支付(内购)。 第三方支付包括:支付宝支付、微信支付、银联支付、...
    帅不过oneS阅读 2,843评论 2 8
  • iOS支付 iOS支付分为两类,第三方支付和应用内支付(内购)。 第三方支付包括:支付宝支付、微信支付、银联支付、...
    sillen阅读 1,351评论 0 1
  • 1 母亲说,当你真正的喜欢上一个男人的时候,只要他有担当,有血性,那么你就不要去衡量他现有的财富值是否配得上你的爱...
    book君阅读 511评论 0 2
  • 和师师认识快两天了,都没怎么说过话。 她邀我一起去放孔明灯,拖着从茶山下来还没缓解的疲惫,随之。 距离澜沧江边还有...
    王小北0206阅读 568评论 2 1