iOS支付宝(Alipay)接入详细流程,比微信支付更简单,项目实战中的问题分析
最近在项目中接入了微信支付和支付宝支付,总的来说没有那么坑,很多人都说文档不全什么的,确实没有面面
俱到,但是认真一步一步测试下还是妥妥的,再配合懂得后台,效率也是很高的,看了这篇文章,你也只要几分钟,
就能轻松接入支付宝,在别人投来崇拜的眼光的同时,你就能潇洒的回一句,略懂略懂。。。。。。
先给大家我写的微信支付,很详细哦,喜欢的点个赞点击打开微信支付链接
前期准备
首先商户要去支付宝申请,得到一个parnter和seller,然后在电脑上生成一组RSA,公钥给支付宝,私钥自己流程,具
体流程我不BB了,自己看点击打开链接,主要这些东西我们App也不需要怎么管理,大家随意看看知道下
1.seller ID 商户唯一ID标识符
2.parnter ID 合作身份ID,以2088开头的16位纯数字
3.加密用到的文件(私钥自己做,支付宝公钥)
下载SDK
这里SDK的文档只有最近一次更新的,没记错的话2016年Alipay有一次重大的改动,我这有一份很全的更新文档,朋
友给我的,需要注意的在最后给大家分析,这里只需要先下载下来,看看里面的代码,然后把需要的东西准备下,继
续往下分析,其实还有一个用处,就是把你生成的密钥和seller和parnter给填进去,跑一下,看能不能拉起来,但是
正常情况下是不需要这么搞得,后台处理的,项目的时候也不需要我们App做的,所以还是往下看
点击下载SDK链接
手动接入SDK
为什么手动呢,因为不支持cocoapods,又不是第三方的,这种还是自己接入安全点。
步骤1
先把这两个框架拖进来
步骤2
把四个箭头的文件拖进去,记得文件夹拖进去是选择Create Group,其实这四个文件是为了签名在客户端签名而存在
的,实际项目中,我们是要把私钥保存在自己的服务器,然后服务器进行支付订单参数的签名,返回给客户端,我还
是放到自己项目中了,毕竟其他地方也要用到,要是在服务器签名order文件就不要拉进来了
步骤3
配置prefix header
如果原本项目中的根目录中有以下prefix文件,例如
则在其中引入 #import<UIKit/UIKit.h>,#import<Foundation/Foundation.h>
如果不存在,那么直接把支付宝Demo里面的prefix文件拖到自己项目的根目录下,最好改下文件名,然后配置路径
(在Build Settings 中搜索 Prefix Header),添加路径 (可以直接拖拽)
步骤4
然后根据下图添加库文件
貌似这两个也要加上,那就加进去吧
步骤5
如果是自己Demo本地签名,不是服务器签名,那么再设置下openssl的路径
在header search paths 中添加 openssl 的路径$(SRCROOT)/XXX
$(SRCROOT)就是本工程的文件夹
网上找了个相对路径和绝对路径的文章点击打开链接
步骤6
配置URL Scheme,主要就是回调的时候能拉起App,这里的Scheme千万不能乱用,也别重复了,不然怎么都拉
不起来了,我就遇到过一次,和微信拉起App重复了,导致很尴尬,咱们保持唯一就好了,也就是给咱们自己的项目
增加个标识符,能够让支付宝回调拉起来
步骤7
这里就是根据上面的配置,在本地进行签名,拉起支付宝,需要的朋友直接下载Demo看吧,没什么好讲的,因为支
付宝都强调了不要在本地玩,不过你想玩,那就玩吧,后果自负
来来来,组装完毕,正式项目接入流程点火起飞
如果起飞有问题,可能我上面漏了什么,记得留言给我哦(写代码记得包头文件)
第一步
传订单号给自己的服务器签名,再提一次,千万别再本地写,你和后台定好规则,传需要的订单字符串给他,他把签
名之后的字符串给你
// 订单的“,”分割字符串 支付宝只是传签名,不需要再次校验,微信成功之后没有返回值,所以微信支付需要保存到沙盒
@property (nonatomic,strong)NSString *orderStr;
关键代码,拉取服务器的签名参数,然后传给支付宝拉起支付 (这里的block处理的是没有支付宝的时候走H5收银台的回调)
- - (IBAction)payMoney:(UIButton *)sender
- {
- // 配置好的回调scheme
- NSString *appScheme = @"上面配置好的";
- // 订单拿去给服务器签名
- [[TWTShoppingCartLogic sharedData] goToSignOrder:@"订单字符串" way:@"1" complete:^(NSError *error, id data) {
- if (error)
- {
- DSToast *toast = [[DSToast alloc] initWithText:@"连接服务器失败,请稍后再试" color:RGBA(255, 174, 1, 1)];
- [toast show];
- }
- else
- {
- // 拉起支付宝,2016年5月之后更新是H5回调到这里
- [[AlipaySDK defaultService] payOrder:data fromScheme:appScheme callback:^(NSDictionary *resultDic) {
- DDLogVerbose(@"点击支付获取的直接结果===>>>>>>reslut = %@",resultDic);
- // NSString *memo = resultDic[@"memo"];
- NSString *result = resultDic[@"result"];
- DDLogVerbose(@"%@",result);
- NSString *resultStatus = resultDic[@"resultStatus"];
- //9000 订单支付成功 正常流程会进入这里 如果中断了就去外面delegate那里的Block
- if ([resultStatus isEqualToString:@"9000"])
- {
- // "out_trade_no" = "\"74db120f0a8e5646ef5a30154e9f6deb\"";
- NSString *outNum = nil;
- NSMutableString *outTradeNum = [[NSMutableString alloc] init];
- NSArray *strArr = [result componentsSeparatedByString:@"&"];
- for (NSString *trade in strArr) {
- if ([trade hasPrefix:@"out_trade_no"])
- {
- outNum = [trade componentsSeparatedByString:@"="][1];
- [outTradeNum appendString:outNum];
- [outTradeNum replaceOccurrencesOfString:@"\"" withString:@"" options:NSLiteralSearch range:NSMakeRange(0, outTradeNum.length)];
- DDLogVerbose(@"%@",outTradeNum);
- break;
- }
- }
- // 第一个参数传的是alipay返回给我的out_trade_id 异步回调二次验证
- [[TWTShoppingCartLogic sharedData] goToCheckOrder:outTradeNum complete:^(NSError *error, id data) {
- if (error)
- {
- [TWTAlertView showAlertViewWithTitle:nil message:@"网络数据异常,请稍后刷新" cancelOnTouch:NO cancelButtonTitle:nil doneButtonTitle:@"确定" doneBlock:^(void){
- [self.navigationController popViewControllerAnimated:YES];
- }];
- // 匹配不到就上报异常
- }
- else
- {
- [[TWTCommonAPILogic sharedData]reportData:nil forEventKey:@"PaySucceedVC_CallBack" number:@(1)];
- DSToast *toast = [[DSToast alloc] initWithText:@"支付成功" color:nil];
- [toast show];
- }
- }];
- }
- else
- {
- //8000 正在处理中 4000 订单支付失败 6001 用户中途取消/重复操作取消 6002 网络连接出错 ---> 从H回来
- // 弹出失败的界面,弹出之后的话就不支付界面消失,方便重新支付
- }
- }];
- }
- }];
- }
注意啦!!!
注意啦!!!
注意啦!!!
介绍下支付宝对我来说影响最大的一次
首先看看官方的调用接口,根据图来讲,我们就说方法1和方法2了,各位懵逼的就看图指示
再来看看2016年5月份更新的文档
iOS版本号:v15.1.0
修改时间:2016-4-28
更新点:
1. 增加获取trade_token接口(收单收银分离参数,与现有业务支付不冲突);
2. 支付结果返回统一,便于用户接入。
原有方案:payOrder:fromScheme:callback接口,业务传入callback1,用于支付结果返回;
processOrderWithPaymentReulst:standbyCallback接口,业务传入callback2,用于支付结果返回;
一般一次支付中(跳支付宝客户端支付),都会涉及到这两个接口;原有的规则是callback1存在,则统一调用callback1返回支付结果给业务方,若callback1不存在(app被杀掉的情况),则调用callback2返回支付结果给业务方。
现有方案:payOrder:fromScheme:callback接口,业务会传入CompletionBlock1(本地没有安装支付宝客户端的情况下(走H5收银台),会通过该CompletionBlock1反馈结果);
processOrderWithPaymentResult:standByCallback接口,业务会传入CompletionBlock2(本地安装了支付宝客户端的情况下,会通过该CompletionBlock2反馈结果)。
个人分析:
改版之前
看了这两次的介绍和图,简单来说就是原本方法1和方法2实现的情况下,方法1存在的情况下,结果返回给方法1的
block,方法1不存在,那么久回调到方法2的回调
唯一存在的用户操作就是,当你支付拉起支付宝的时候,你的商户App自杀了或者你手贱把他退出了,那么,你的方
法1自然就没了,你就需要方法2来进行回调结果的通知
改版之后
我感觉更正常了点了,但是咱们得改改代码逻辑啊,方法1和方法2都实现了,方法1只负责当没有支付宝客户端的时
候,走H5收银台,进行支付,结果处理回调,方法2就处理有客户端的情况下的回调,这里亲自测试,当拉起支付宝
的时候,你手贱退出了,也是回调到方法2的block里面进行结果反馈
我已经更新SDK到最新了,而且我这里介绍的也是最新,就是改版之后的方法,老的不介绍了,理论上都要用最新的
嘛,如果你用旧的,那我也没办法
第二步
处理支付结果(有支付宝的情况下回调该方法)
- // 当用户通过其它应用启动本应用时,会回调这个方法,url参数是其它应用调用openURL:方法时传过来的
- - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation
- {
- if ([url.host isEqualToString:@"safepay"]) {
- //跳转支付宝钱包进行支付,处理支付结果,该方法是当去支付的时候我的进程被杀死的时候调用这里的回调,不然就调用payorder的回调
- [[AlipaySDK defaultService] processOrderWithPaymentResult:url standbyCallback:^(NSDictionary *resultDic)
- {
- DDLogVerbose(@"点击支付获取的直接结果===>>>>>>reslut = %@",resultDic);
- NSString *result = resultDic[@"result"];
- NSString *resultStatus = resultDic[@"resultStatus"];
- //9000 订单支付成功
- // 支付完成需要把返回的outtradeID给解析出来,给服务器二次确认
- if ([resultStatus isEqualToString:@"9000"])
- {
- // "out_trade_no" = "\"74db120f0a8e5646ef5a30154e9f6deb\"";
- NSString *outNum = nil;
- NSMutableString *outTradeNum = [[NSMutableString alloc] init];
- NSArray *strArr = [result componentsSeparatedByString:@"&"];
- for (NSString *trade in strArr) {
- if ([trade hasPrefix:@"out_trade_no"])
- {
- outNum = [trade componentsSeparatedByString:@"="][1];
- [outTradeNum appendString:outNum];
- [outTradeNum replaceOccurrencesOfString:@"\"" withString:@"" options:NSLiteralSearch range:NSMakeRange(0, outTradeNum.length)];
- break;
- }
- }
- order.aliPayTrade = outTradeNum;
- // 第一个参数传的是alipay返回给我的out_trade_id 二次确认
- [[TWTShoppingCartLogic sharedData] goToCheckOrder:outTradeNum complete:^(NSError *error, id data) {
- if (error)
- {
- [TWTAlertView showAlertViewWithTitle:nil message:@"网络数据异常,请稍后刷新" cancelOnTouch:NO cancelButtonTitle:nil doneButtonTitle:@"确定" doneBlock:^(void){
- }];
- // 匹配不到就上报异常
- }
- else
- {
- [[TWTCommonAPILogic sharedData]reportData:nil forEventKey:@"PaySucceedVC_CallBack" number:@(1)];
- DSToast *toast = [[DSToast alloc] initWithText:@"支付成功" color:nil];
- [toast show];
- }];
- }
- else
- {
- //8000 正在处理中 4000 订单支付失败 6001 用户中途取消/重复操作取消 6002 网络连接出错
- }
- }];
- }
- return YES;
- }
这里有个iOS 9的API,反正没什么卵用
- // NOTE: 9.0以后使用新API接口
- - (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<NSString*, id> *)options
- {
第三步(小细节)
和微信支付一样,我们不能直接用同步返回的结果来判断是否正真支付成功(也是我写的微信支付详解链接)
按照官方的说法请看如下
好在支付宝和微信支付不同的地方在于,支付宝的返回数据中有带订单信息,我们就直接解析可以了,不需要像微信
一样只返回状态码,我们需要本地存起来,再和服务器二次校验
几个小小的坑
Cannot find interface declarationfor 'NSObject', superclass of'Base64'
‘rsa.h’ file not found
写的比较仓促,基本逻辑是写好了,注意的事项也差不多,如有漏的,或者不同意见的
请留言告诉我哦
有帮到你的帮忙点个赞,各位的认可是我分享的最大动力
iOS支付宝(Alipay)接入详细流程,比微信支付更简单,项目实战中的问题分析相关推荐
- Android App支付:支付宝SDK接入详细指南(附官方支付demo)
Android App支付:支付宝SDK接入详细指南(附官方支付demo) 前言 一家移动互联网公司,说到底,要盈利总是需要付费用户的,自己开发支付系统对于资源有限的公司来说显然不太明智,国内已经有多 ...
- Android App支付系列(二):支付宝SDK接入详细指南(附官方支付demo)
前言 一家移动互联网公司,说到底,要盈利总是需要付费用户的,自己开发支付系统对于资源有限的公司来说显然不太明智,国内已经有多家成熟的移动支付提供商,阿里就是其中之一. 继< Android Ap ...
- 支付宝SDK接入详细指南(附官方支付demo)
前言 一家移动互联网公司,说到底,要盈利总是需要付费用户的,自己开发支付系统对于资源有限的公司来说显然不太明智,国内已经有多家成熟的移动支付提供商,阿里就是其中之一. 继< Android A ...
- Android App支付系列(一):微信支付接入详细指南(附官方支付demo)
写在前面 一家移动互联网公司,说到底,要盈利总是需要付费用户的,自己开发支付系统显然是不明智的,国内已经有多家成熟的移动支付提供商,腾讯就是其中之一.梳理了下微信支付的接入,今天给大家分享下腾讯旗下的 ...
- IOS应用开发-发布详细流程
IOS应用开发-发布详细流程 申请成为IOS开发者(1天) https://connect.apple.com/cgi-bin/WebObjects/register.woa/wa/default?u ...
- Hbuilder集成微信支付教程(简单流程)
整个系统运作的流程大致是这样: (1)APP向服务器发出需要付款的请求 (2)服务器请求微信服务端下单 (3)服务器将下单数据回复给APP (4)APP向微信服务端发起请求并输入密码进行支付 关键字用 ...
- 视频教程-19年录制SpringBoot2.x整合微信支付在线教育网站项目实战-Java
19年录制SpringBoot2.x整合微信支付在线教育网站项目实战 7年的开发架构经验,曾就职于国内一线互联网公司,开发工程师,现在是某创业公司技术负责人, 擅长语言有node/java/pytho ...
- 支付宝app支付流程(微信支付同理)
支付宝app支付现在很方便,支付宝的参考文档也规范易懂,需要开发人员做的事很少,具体参考支付宝开发文档https://docs.open.alipay.com/204/105297/ ,后台服务端的话 ...
- 微信支付服务器验证的java_Java中的微信支付(3):API V3对微信服务器响应进行签名验证...
1. 前言 牢记一句话:公钥加密,私钥解密:私钥加签,公钥验签. 微信支付V3版本前两篇分别讲了如何对请求做签名和如何获取并刷新微信平台公钥,本篇将继续展开如何对微信支付响应结果的验签. 2. 为什么 ...
最新文章
- 会计基础第一章模拟试题(1)
- 【Java】泛型中 extends 和 super 的区别?
- c语言三目运算错误,c语言中三目运算符有什么用
- 厦门one_“断轴”频发,李想承认理想ONE存缺陷!曾声明悬架非常安全
- 异常处理-RestFul的异常处理
- OSGI(面向Java的动态模型系统)和它的实现Equinox
- 【华为云技术分享】深度详解GaussDB bufferpool缓存策略
- Scala学习之Option类
- QT5.14在Win10和Win7下的安装
- Yii DataProvider
- linux 软件应用
- html的基本标记符,html的基本标记符号
- android 一位小数_android如何保留小数点后x位数字
- jmeter下载地址
- xposed框架安装使用步骤
- 步进电机 高速光耦_光耦的参数以及高速光耦如何选型!-先进光半导体
- 【Swift】SpotLight搜索
- linux修改mac地址_如何(以及为什么)在Windows,Linux和Mac上更改您的MAC地址
- IntelliJ IDEA 最新注册码(截止到2019年12月12日)
- 【数据结构复习】二叉树的遍历——从微软2014校园招聘说起
热门文章
- Linux下的内存对齐函数
- 【TX2】英伟达Nvidia TX2连接蓝牙设备
- 【Linux】neocomplcache disabled: “sudo vim“ is detected and $HOME is set to your user‘s home
- 【Qt】通过QtCreator源码学习Qt(六):命令行参数解析实现
- 海思3536:交叉编译Qt4.8.4
- Ruby之Enumerator类
- mysql使用sha256密码,MySQL5.6启用sha256_password插件
- Java项目:医院分诊挂号住院管理系统(java+SpringBoot+FreeMarker+Mysql)
- c语言作业扩展名通常为什么,C语言的源程序通常的扩展名是( )
- 管理 zabbix_Zabbix 2019 峰会丨看睿象云如何在 Zabbix 中玩转告警