为了方便项目使用,我将支付封装成了一个工具类,提供整个项目使用。
这里用一个类似支付的例子,假设在一个支付界面,用户点击了支付按钮发起了支付。这个支付界面我们起名为MPayController。
第一步:发起支付
MPayController.m
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
UIButton *payBtn = [UIButton buttonWithType:UIButtonTypeRoundedRect];
[payBtn setTitle:@"支付" forState:UIControlStateNormal];
[payBtn setFrame:CGRectMake(16, CGRectGetHeight(self.view.frame)-16, CGRectGetWidth(self.view.frame)-32, 44)];
[payBtn addTarget:self action:@selector(payAction:) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:payBtn];
}
用户点击这里的支付,发起App支付请求,点击按钮事件如下:
- (void)payAction:(UIButton *)payBtn {
/**
发起支付宝支请求取前,需要组装好请求信息。这一步应在商户服务端完成,商户服务端直接将组装和签名后的请求串orderString传给客户端,客户端直接传给SDK发起请求。
*/
/**
支付宝支付简化成两个步骤:
步骤一:支付
调起支付到支付,需要三步:
第一步:客户端点击发起支付,请求服务端接口,服务端直接将组装和签名后的请求串orderString传给客户端;
第二步:客户端拿到返回数据,调起支付宝支付SDK;
第三步:跳转支付宝钱包(支付网页)进行支付,处理支付结果
步骤二:支付校验
同步返回支付结果为9000后需要做支付校验:
客户端拿到支付宝的支付结果后,提取相应的字段发送给服务端,服务端校验通知参数的合法性,只有全部验证通过后,才可以认定买家付款成功。ps:支付成功后支付宝会异步发送通知给服务端。
*/
//第一步:客户端点击发起支付,请求服务端接口,服务端直接将组装和签名后的请求串orderString传给客户端;
[MPayModel requestPayInfimationWithOrderID:@"xxx" sucess:^(NSDictionary *result) {
//请求成功,调支付宝支付SDK
if ([result[@"code"] integerValue] == 200)
{
//第二步:客户端拿到返回数据,调起支付宝支付SDK;
NSString *orderString = result[@"responseBody"][@"orderString"];
[[MPayManager sharedManager] alipayWithOrderString:orderString];
}
//请求失败
else {
}
} failure:^{
}];
}
这里MPayModel是一个普通的model,用来请求服务端接口并处理回调。MPayManager是我们封装的支付管理类。
MPayModel.h
#import <Foundation/Foundation.h>
@interface MPayModel : NSObject
/**
用订单id请求服务端支付宝支付的组装信息
@param oid 订单id
@param sucess 成功回调,result里面有服务端组装和签名后的请求串orderString
@param failure 失败回调
*/
+ (void)requestPayInfimationWithOrderID:(NSString *)oid
sucess:(void(^)(NSDictionary *result))sucess
failure:(void(^)(void))failure;
@end
MPayModel.m
#import "MPayModel.h"
@implementation MPayModel
+ (void)requestPayInfimationWithOrderID:(NSString *)oid
sucess:(void(^)(NSDictionary *result))sucess
failure:(void(^)(void))failure {
//这里模拟请求接口返回服务端组装和签名后的字符串orderString和状态码
NSDictionary *resultDict = @{@"code":@"200",
@"responseBody":@{@"orderString":@"xxx"},
};
if (sucess) {
sucess(resultDict);
}
}
@end
因为我们只是写个小demo,没有测试服务器,所以接口模拟返回orderString。
MPayManager.h
#import <Foundation/Foundation.h>
@interface MPayManager : NSObject
+ (instancetype)sharedManager;
#pragma mark -
#pragma mark - Alipay
/**
调起支付宝支付SDK
@param orderString 服务端已组装和签名后的请求串orderString
*/
- (void)alipayWithOrderString:(NSString *)orderString;
/**
支付宝客户端返回url处理方法,跳转支付宝钱包进行支付,处理支付结果
@param url 支付宝url(如果客户端已安装支付宝App则直接跳转App,否则跳转支付宝支付网页)
*/
- (void)alipayOpenUrl:(NSURL *)url;
在AppDelegate里面需要处理,才能打开支付宝钱包或者支付宝支付网页
- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation {
//第三步(<iOS9):跳转支付宝钱包(支付网页)进行支付,处理支付结果
if ([url.host isEqualToString:@"safepay"]) {
[[MPayManager sharedManager] alipayOpenUrl:url];
}
return YES;
}
// NOTE: 9.0以后使用新API接口,iOS9之前还是走上接口,所以两个方法中都需要处理
- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<UIApplicationOpenURLOptionsKey,id> *)options {
//第三步(>=iOS9):跳转支付宝钱包(支付网页)进行支付,处理支付结果
if ([url.host isEqualToString:@"safepay"]) {
[[MPayManager sharedManager] alipayOpenUrl:url];
}
return YES;
}
第二步:支付校验
先看看MPayManager的实现文件:
#import "MPayManager.h"
#import <AlipaySDK/AlipaySDK.h>
@implementation MPayManager
+ (instancetype)sharedManager {
static MPayManager *instance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
instance = [[self alloc] init];
});
return instance;
}
#pragma mark -
#pragma mark - Alipay
- (void)alipayWithOrderString:(NSString *)orderString
{
//appScheme为app在info.plist注册的scheme
NSString *appScheme = @"xxxx";
[[AlipaySDK defaultService] payOrder:orderString fromScheme:appScheme callback:^(NSDictionary *resultDic)
{
NSInteger status = [resultDic[@"resultStatus"] integerValue];
switch (status) {
case 9000://订单支付成功
//代理回调支付结果,将支付结果发送至服务器,服务器验签,解析支付结果,全部字段验证通过后,才可以认定买家付款成功。
break;
case 8000://正在处理中,支付结果未知(有可能已经支付成功),请查询商户订单列表中订单的支付状态
break;
case 4000://订单支付失败
break;
case 5000://重复请求
break;
case 6001://用户中途取消
break;
case 6002://网络连接出错
break;
case 6004://支付结果未知(有可能已经支付成功),请查询商户订单列表中订单的支付状态
break;
}
}];
}
- (void)alipayOpenUrl:(NSURL *)url {
//第三步:跳转支付宝钱包(支付网页)进行支付,处理支付结果
//跳转支付宝钱包进行支付,处理支付结果
[[AlipaySDK defaultService] processOrderWithPaymentResult:url standbyCallback:^(NSDictionary *resultDic) {
NSLog(@"pay result = %@",resultDic);
}];
//授权跳转支付宝钱包进行支付,处理支付结果(跳转支付宝钱包的网页支付)
[[AlipaySDK defaultService] processAuth_V2Result:url standbyCallback:^(NSDictionary *resultDic) {
NSLog(@"result = %@",resultDic);
// 解析 auth code
NSString *result = resultDic[@"result"];
NSString *authCode = nil;
if (result.length>0) {
NSArray *resultArr = [result componentsSeparatedByString:@"&"];
for (NSString *subResult in resultArr) {
if (subResult.length > 10 && [subResult hasPrefix:@"auth_code="]) {
authCode = [subResult substringFromIndex:10];
break;
}
}
}
NSLog(@"授权结果 authCode = %@", authCode?:@"");
}];
}
@end
最后给MPayManager写一个代理,当收到状态码为9000后,回调,然后将resultDic里面的某些字段发送与服务端,服务端做校验,校验成功才算支付成功。
个人觉得只是出来9000还不够,当状态码为9000、8000或6004时候都需要回调就行校验?成功不成功的就知道了。
最后就是支付成功或者失败界面上的显示了。