微信支付原理及实现

原理图:

微信支付原理及实现_第1张图片
屏幕快照 10.47.58.png

1.微信开放平台账号的注册
微信开放平台(微信支付需要付费的,注册成功后使用的主要为AppKey/SecretKey,其中SecretKey交由后台完成集成)
微信开放平台是商户APP接入微信支付开放接口的申请入口,通过此平台可申请微信APP支付
平台入口: http://open.weixin.qq.com
2.项目的修改与设置
1)首先应该下载SDK并导入项目(这里以友盟自带微信支付说明,具体内容相似)
如果项目中使用了友盟社会化分享SDK,那么你将不需要重复导入SDK,友盟中已经集成了微信支付。具体SDK如下:
微信支付原理及实现_第2张图片
屏幕快照 10.53.41.png

如果你的项目没有使用友盟,请到开放平台下载最新的SDK,然后导入您的项目。具体的SDK如下:
屏幕快照 10.56.05.png

2)在添加完SDK后需要添加相应的库以及文件
微信支付原理及实现_第3张图片
屏幕快照10.57.24.png

3)然后是添加对应的URL用于调起
微信支付原理及实现_第4张图片
屏幕快照10.59.14.png

4)添加白名单、关闭Bitcode(对于http/https适配iOS9.0自行修改)
微信支付原理及实现_第5张图片
屏幕快照11.00.47.png

3.SDK的注册与回调
在完成了上面的操作之后,我们所导入的SDK就可以使用了,很简单
1)首先我们要进行注册
导入#import “WXApi.h”

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { 
// Override point for customization after application launch. 
ViewController *userGuideViewController = [[ViewController alloc] init]; 
self.window.rootViewController = userGuideViewController;
 [WXApi registerApp:@"这里添加APPKey"]; 
return YES;
}

2)接着我们要写一个调起

- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary *)options{
 // 跳转到URL scheme中配置的地址 //
NSLog(@"跳转到URL scheme中配置的地址-->%@",url);
 return [WXApi handleOpenURL:url delegate:(id)self];
}

3)然后是回调

//支付成功时调用,回到第三方应用中
- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation{
 // NSLog(@"****************url.host -- %@",url.host); 
if ([url.scheme isEqualToString:@"这里添加AppKey"]) { 
return [WXApi handleOpenURL:url delegate:(id)self];
 } 
return YES;}
- (void) onResp:(BaseResp*)resp{ 
NSString *strMsg = [NSString stringWithFormat:@"errcode:%d", resp.errCode];
 NSString *strTitle = @"支付结果";
 if([resp isKindOfClass:[PayResp class]]){ 
switch (resp.errCode) { 
case WXSuccess:
 strMsg = @"支付结果:成功!"; 
break; 
case WXErrCodeUserCancel: 
strMsg = @"支付结果:用户点击取消!";
 break; 
case WXErrCodeSentFail: 
strMsg = @"支付结果:发送失败!"; 
break; 
case WXErrCodeAuthDeny:
 strMsg = @"支付结果:授权失败!";
 break; 
default: strMsg = @"支付结果:微信不支持!";
 break;
 } 
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:strTitle message:strMsg delegate:self cancelButtonTitle:@"确定" otherButtonTitles:nil, nil];
 [alert show]; 
}
}

4.SDK的使用
1)最后是在项目中需要进行微信支付的地方使用支付(这是完成整个支付流程中最重要的一部分,也是最需要注意的地方)
首先要进行一次网络请求,后台接口,将你的商品信息发送到后台,生成订单后返回,返回信息中最少要包含六个参数,通过这些参数调起支付。
导入#import “WXApi.h”

//返回参数调起支付
PayReq* req = [[PayReq alloc] init]; 
NSMutableDictionary *dict=[result objectForKey:@"resultCode"]; 
req.openID = [dict objectForKey:@"appid"]; req.partnerId = [dict objectForKey:@"mch_id"];
 req.prepayId = [dict objectForKey:@"prepay_id"]; 
req.nonceStr = [self md5:time_stamp]; 
req.timeStamp = [time_stamp intValue];
 req.package = @"Sign=WXpay"; 
//签名(这个签名可以自己进行签名,也可以后台操作,由后台返回) 
NSMutableDictionary *signParams=[[NSMutableDictionary alloc] init]; 
[signParams setObject: req.openID forKey:@"appid"]; 
[signParams setObject: req.nonceStr forKey:@"noncestr"]; 
[signParams setObject: req.package forKey:@"package"]; 
[signParams setObject: req.partnerId forKey:@"partnerid"]; 
[signParams setObject: time_stamp forKey:@"timestamp"]; 
[signParams setObject: req.prepayId forKey:@"prepayid"]; 
req.sign = [self createMd5Sign:signParams];
 //调起支付(**参数有一个错误,将不能完成调起**) 
[WXApi sendReq:req];
  1. 自行加密处理(这部分可以由后台完成)
//微信支付加密
-(NSString *) md5:(NSString *)str{ 
const char *cStr = [str UTF8String]; 
unsigned char digest[CC_MD5_DIGEST_LENGTH]; 
CC_MD5( cStr, (unsigned int)strlen(cStr), digest ); 
NSMutableString *output = [NSMutableString stringWithCapacity:CC_MD5_DIGEST_LENGTH * 2]; 
for(int i = 0; i < CC_MD5_DIGEST_LENGTH; i++) 
[output appendFormat:@"%02X", digest[i]]; 
return output;
}
-(NSString*) createMd5Sign:(NSMutableDictionary*)dict{ NSMutableString *contentString =[NSMutableString string]; 
NSArray *keys = [dict allKeys]; 
//按字母顺序排序 
NSArray *sortedArray = [keys sortedArrayUsingComparator:^NSComparisonResult(id obj1, id obj2) { 
return [obj1 compare:obj2 options:NSNumericSearch]; 
}]; 
//拼接字符串 
for (NSString *categoryId in sortedArray) { 
if (![[dict objectForKey:categoryId] isEqualToString:@""] && ![categoryId isEqualToString:@"sign"] && ![categoryId isEqualToString:@"key"] ) {
 [contentString appendFormat:@"%@=%@&", categoryId, [dict objectForKey:categoryId]]; 
} 
}
 //添加key字段 
[contentString appendFormat:@"key=%@", @"这里添加SecretKey"]; 
NSString *md5Sign =[WXUtil md5:contentString]; return md5Sign;}

5.常见异常Bug处理
1)常见问题为微信支付调起不成功,每次调起只能调起一个带有返回确认按钮的页面。
原因:参数错误,特别注意参数(签名),后台返回的参数不正确。
解决方案:逐个校验参数,签名可以本地加密处理,不要依赖后台的返回签名,有的时候后台返回的是不正确的。

你可能感兴趣的:(微信支付原理及实现)