关于内购所需东西: 
1.测试开发证书:需要打开in-app-purchase,绑定bundleid:com.aragon.TexasPoker 
2.iTunes connect 里添加内购应用: 
1>首先需要创建一个app。套装id 选择证书里的创建好的这个项目app id 
2>功能里添加app 内购买项目:每个内购买项目的id必须唯一,id需要测试期间先发给我,之后发给后台,有后台返给我(选择消耗型的) 
3>填写协议、税务和银行业务,填写所有的信息,(网上有文档) 
4>用户与职能里添加一个沙盒测试员,邮箱密码发给我,我需要这个做内购买支付测试,

关于内购

前期准备任务 
1>配置开发测试证书: 
前往苹果开发者网站(https://developer.apple.com)使用开发者账号登入,进入accout页面


选择certifications,Identitiers,&profiles

如果你已经有证书了,就不用创建证书了(证书不需要很多个,基本上两个证书就足够了,一个开发证书一个测试证书),直接创建app ids 
点击app ids 点击“+”,按照里边的步骤添加name, 和bundle id,注意打开In-App-purchase,需要和应用的bundleid 对应,很简单 
然后创建描述文件

 
选择dev ,继续

 
选择刚才创建的app id,继续 
然后选择证书,继续 
之后选择测试的设备,继续,创建描述文件结束,可以将这个描述文件导出,之后用 
2>配置工程

 
去掉自动配置,import刚才导出的描述文件,如果你安装过证书他会自己识别,如果没有安装过,那么需要把证书导出来,双击安装 
3>Itunes Connect 添加应用(这也是内购应用比较重要的一步): 
1>创建内购app,和内购项目 
1:到苹果开发者网站accout页面(或者直接进入iTunes connect 网站登录账号,需要苹果开发者账号) 
2:
选择“我的app”进入添加内购项目

 
点击右上角新建app

 
选择平台,名称,注意套装id选择我们那会在创建证书里的App IDs,sku 随便填,点击创建,这时候就会创建出一个新的app 
3:点击刚新创建的app

 
选择功能,


添加内购项目点击+

 
选择一种类型(上边的解释很清楚,根据需要选择)创建之后填写基本的信息


需要注意的是产品id一定唯一,这个id我们可以放在后台,需要的时候返回给我们,如果你的内购项目只有一个也可以放在前台,填写完成之后,选择保存,创建内购项目完成 
2>创建沙盒测试 
点击“用户和职能”

 
选择沙箱技术测试 
点击“+” 
填写基本信息

注意电子邮箱随便写一个,但是不要求是真的,选择储存,创建完毕,这个账号是以后测试需要的apple id 
3>填写税务和银行卡信息 
选择“税务”这一块,填写基本的信息,这块一般是功能的产品填的,太机密,我就没有权限看了 
所有的前奏准备完成,现在开始代码部分

—————————————————————————— 
我们用到的这个框架是:StoreKit  
先说一下,内购的基本思路: 
1>我们需要判断是否支持内购 
2>我们需要通过产品id(也就是我们创建内购Item的那个产品id)去获取更全面的info 
3>返回全面的产品info之后,我们将这个产品加入到支付队列里等待支付 
4>系统通过通知告诉我们现在的状态 
5>当支付成功之后,我们需要通过沙盒存储的Url,获取Data 
6>将Data转化base64加密的字符串,提交到后台,后台去验证信息,如果验证成功,这时候,支付才是真正的完成了

下面直接复制上我写的demo,代码

//
// ViewController.swift
// IN_App_Purchase_Demo
//
// Created by HaoYuhong on 2017/9/7.
// Copyright © 2017年 HaoYuhong. All rights reserved.
//
 
import UIKit
import StoreKit
import SVProgressHUD
 
// testting
 
let Diamond60 = "ACS_Diamond_60"
class ViewController: UIViewController,SKProductsRequestDelegate, SKPaymentTransactionObserver {
/// IN-APP Purchase
var productRequest:SKProductsRequest?
var currentPaymentItemID:String?
override func viewDidLoad() {
super.viewDidLoad()
SKPaymentQueue.default().add(self)
}
 
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
@IBAction func payAction(_ sender: Any) {
buyDiamond()
}
 
 
}
//MARK:- In-APP Purchase
extension ViewController {
 
func buyDiamond() {
if SKPaymentQueue.canMakePayments() {
currentPaymentItemID = Diamond60
self.requestProductData(id: Diamond60)
} else {
self.showAlert("", "您的手机未开启允许内购服务", "确定")
}
SVProgressHUD.show()
}
func requestProductData(id:String) {
let productIdSet = NSSet(object: id)
productRequest = SKProductsRequest(productIdentifiers: productIdSet as! Set<String>)
productRequest?.delegate = self
productRequest?.start()
}
 
//MARK:- Product Request Delegate
 
func productsRequest(_ request: SKProductsRequest, didReceive response: SKProductsResponse) {
let products = response.products
if products.count == 0 {
self.showAlert("", "未找到你所该买的产品", "确定")
SVProgressHUD.dismiss()
return
}
var paymentProduct:SKProduct?
for product in products {
print(product.localizedDescription)
print(product.price)
print(product.productIdentifier)
if currentPaymentItemID! == product.productIdentifier {
paymentProduct = product
}
}
if let p = paymentProduct {
let payment = SKPayment(product: p)
SKPaymentQueue.default().add(payment)
}else {
SVProgressHUD.dismiss()
}
}
func request(_ request: SKRequest, didFailWithError error: Error) {
SVProgressHUD.dismiss()
self.showAlert("", "购买失败", "确定")
}
 
func requestDidFinish(_ request: SKRequest) {
print("requestDidFinish")
}
 
//MARK:- Transition Observe
 
func paymentQueue(_ queue: SKPaymentQueue, updatedTransactions transactions: [SKPaymentTransaction]) {
for transaction in transactions {
switch transaction.transactionState {
case .purchased:
print("交易完成")
self.completeTransaction(transaction)
SKPaymentQueue.default().finishTransaction(transaction)
case .purchasing:
print("商品添加进列表")
case .failed:
print("交易失败")
self.failedTransaction(transaction)
case .restored:
print("已经购买过此商品")
restoredTransaction(transaction)
default:
break
}
}
}
func completeTransaction(_ transaction:SKPaymentTransaction) {
print("交易结束")
let productId = transaction.payment.productIdentifier
if !(productId.isEmpty) {
// 通过存在沙盒的Url获取Data 提交到自己服务器验证
RequestManager.request((Bundle.main.appStoreReceiptURL?.absoluteString)!, method: .post).responseData(completionHandler: { (data) in
let transactionReceiptData = data.data
let transactionReceiptString = transactionReceiptData?.base64EncodedString(options: Data.Base64EncodingOptions.endLineWithLineFeed)
// 提交自己服务器验证
self.commitServer(transactionReceiptString)
 
})
 
}
}
func restoredTransaction(_ transaction:SKPaymentTransaction) {
 
}
func failedTransaction(_ transaction:SKPaymentTransaction) {
SVProgressHUD.dismiss()
if let nserror = transaction.error as NSError? {
if nserror.code != SKError.Code.paymentCancelled.rawValue {
self.showAlert("", (transaction.error?.localizedDescription)!, "确定")
}
}
SKPaymentQueue.default().finishTransaction(transaction)
}
func commitServer(_ transactionReceiptString: String?) {
if let string = transactionReceiptString {
 
} else {
print("未知错误")
}
}
}

我会将此代码上传到我的github,有需要的可以下载查看https://github.com/HaoXianSen/In_App_Purchase_Demo.git

Apple 内购的更多相关文章

  1. 苹果内购和 Apple Pay

    作者:CC老师_MissCC链接:http://www.jianshu.com/p/e3bc47e81785來源:简书 苹果内购 1.什么是内购? 如果你购买的商品,是在本app中使用和消耗的,就一定 ...

  2. iOS开发系列--通讯录、蓝牙、内购、GameCenter、iCloud、Passbook系统服务开发汇总

    --系统应用与系统服务 iOS开发过程中有时候难免会使用iOS内置的一些应用软件和服务,例如QQ通讯录.微信电话本会使用iOS的通讯录,一些第三方软件会在应用内发送短信等.今天将和大家一起学习如何使用 ...

  3. iOS--通讯录、蓝牙、内购、GameCenter、iCloud、Passbook等系统服务开发汇总

    iOS开发过程中有时候难免会使用iOS内置的一些应用软件和服务,例如QQ通讯录.微信电话本会使用iOS的通讯录,一些第三方软件会在应用内发送短信等.今天将和大家一起学习如何使用系统应用.使用系统服务: ...

  4. [IPA]IOS In App Purchase(内购)验证

    参考我之前的笔记 苹果内购笔记,在客户端向苹果购买成功之后,我们需要进行二次验证. 二次验证 IOS在沙箱环境下购买成功之后,向苹果进行二次验证,确认用户是否购买成功. 当应用向Apple服务器请求购 ...

  5. AppStore ipa (苹果内购)笔记

    内购示意图 准备条件 苹果的开发者证书,已经为应用启用App内购,并在Xcode更新配置文件 itunes store设置 itunes中创建App及其它设置 参考:iOS应用程序内购/内付费(一)  ...

  6. AppStore 内购验证的方法

    AppStore增加了验证内购(In App Purchasement)的方法, 就是苹果提供一个url地址, 开发测试用: https://sandbox.itunes.apple.com/veri ...

  7. iOS开发——高级技术&内购服务

    内购服务 大家都知道做iOS开发本身的收入有三种来源:出售应用.内购和广告.国内用户通常很少直接 购买应用,因此对于开发者而言(特别是个人开发者),内购和广告收入就成了主要的收入来源.内购营销模式,通 ...

  8. iOS开发系列通讯录、蓝牙、内购、GameCenter、iCloud、Passbook系统服务开

    --系统应用与系统服务 iOS开发过程中有时候难免会使用iOS内置的一些应用软件和服务,例如QQ通讯录.微信电话本会使用iOS的通讯录,一些第三方软件会在应用内发送短信等.今天将和大家一起学习如何使用 ...

  9. iOS开发 使用RMStore简化内购代码 + 内购买订单验证

    现在很多的app里面都添加了应用内购买,网上关于苹果证书的生成和设置的教程比较多,这里就不多赘述了,推荐几个个人觉得说的比较详细的网址: http://www.jianshu.com/p/86ac7d ...

随机推荐

  1. 【转载】Linux时间相关结构与函数

    1 时间的获取 在程序当中, 我们经常要输出系统当前的时间,比如日志文件中的每一个事件都要记录其产生时间.在 C 语言中获取当前时间的方法有以下几种,它们所获得的时间精度从秒级到纳秒,各有所不同. 表 ...

  2. tomcat调优(三)

    标签: linux 笔者Q:972581034 交流群:605799367.有任何疑问可与笔者或加群交流 1.安全优化 降权启动 telnet管理端口保护 ajp连接端口保护 禁用管理端 关闭本地默认 ...

  3. 2018-02-02-解决IDE中无法忽略的非代码文件

    layout: post title: 2018-02-02-解决IDE中无法忽略的非代码文件 key: 20180202 tags: GIT 版本管理 modify_date: 2018-02-02 ...

  4. MySQL--当查询遇到隐藏字符

    事件起因: 在将一些EXCEL维护的数据导入MySQL中维护过程中发现漏了一些数据,检查时发现看着相同的SQL返回的结果完全不同: 在SQLyog中看到的截图如: 两个SQL执行返回结果不同,其中一条 ...

  5. Redis和Memcached区别

    本文参考 Redis与Memcached的区别. 如果简单地比较Redis与Memcached的区别,大多数都会得到以下观点: Redis不仅仅支持简单的k/v类型的数据,同时还提供list,set, ...

  6. Python CRM项目四

    实现Django Admin的多对多的复选框效果 效果:左边显示的是未选中的字段,右边显示的是已选中的字段,两边点击的标签可以互相更换 首先在king_admin.py中增加filter_horizo ...

  7. easyui_datagrid 行内使用comobox的编码实现

    easyui datagrid组件的列属性中有一个editor属性,官方介绍如下: 所以,我们可以通过编码实现datagrid行内插入comobox的方式来实现某些场合的需要,具体编码实现如下: // ...

  8. LINUX文件定位

    body, table{font-family: 微软雅黑; font-size: 10pt} table{border-collapse: collapse; border: solid gray; ...

  9. 洛谷 [P1402] 酒店之王

    有两个约束条件的二分图匹配 我们回忆一下二分图匹配的匈牙利算法的具体流程,它是通过寻找增广路来判断最大匹配数的,我们再观察一下题目中的两个条件,只有两个条件都满足,才算找到一条增广路,所以我们可以分别 ...

  10. 利用Needleman–Wunsch算法进行DNA序列全局比对

    生物信息学原理作业第二弹:利用Needleman–Wunsch算法进行DNA序列全局比对. 具体原理:https://en.wikipedia.org/wiki/Needleman%E2%80%93W ...