声明本篇博客为作者原创,本篇是继支付宝支付之后本人又学习的第二种支付实现,本篇着重于原理与注意事项的学习。

参考  参考

微信支付的开发文档相比支付宝的比较简单,但是使用功能丝毫也不含糊,我觉得简单易读的文档是吸引开发者做出喜好选择的第一步。但是个人觉得,微信支付与支付宝的支付的实现思路大致雷同,并不能说是微信另开思路进行支付创新。

微信支付的官方文档中提供了扫码支付、公众号支付、App支付支付模式。开发者要实现用微信支付的功能,需要商户向微信官方申请微信支付权限,商户获得权限后,将支付账户信息告诉开发者。

商户申请的支付账号信息表如下:

好了下面将要却别两个概念,什么是公众号?什么是开放平台?

参考 公众账号就是面向企业、团体、机构、公司、政府等等的拥有一定运营能力一个团结的集合的公众号,用来作为一个自媒体或者服务窗口来用,强调信息流。开放平台是面向那些拥有成熟的应用程序,能够将信息分享给社交圈子的一种偏向技术实现的平台,是面向开发者来说。看看shareSDK中都能够分享到的一些社交平台都是一种开放平台。个人理解。

微信支付的模式有:(参考)刷卡支付、扫码支付、公众号支付、App支付

刷卡支付:(非验密支付)门店收银台生成订单->用户打开客户端生成支付条码->门店扫码->后台生成sign->后台调用刷卡API请求支付->微信支付系统验证支付信息->通过验证处理数据->给商户后台->商户后台对得到的支付结果进行签名验证和处理->{处理成功:微信支付系统会将支付结果返回给商户,同时把支付结果通知给用户(以短信、微信消息的形式通知)}
(验密支付)门店收银台生成订单->向后台发起支付请求->后台调用【提交刷卡支付API】请求支付生成支付交易->微信支付系统验证请求支付信息验证->通过验证要求用户输入密码->微信支付系统返回USERPAYING状态->给商户后台返回收银台应答结果->微支付系统通知用户输入密码->用户输入密码,提交微信支付,万成支付->微信客户端提示微信后台返回在支付结果->用户得到来自微信后台的提示短息->商户收银台得到USERPAYING状态经过商户后台调用【查询订单API】查询实际支付结果->如果支付结果仍为USERPAYING,则每隔5秒循环调用【查询订单API】判断实际支付结果,如果用户取消支付或累计30秒用户都未支付,商户收银台退出查询流程后继续调用【撤销订单API】撤销支付交易。

支付的异常处理:

用户遇到支付异常,请按如下说明处理

(1)用户微信端弹出系统错误提示框,用户可在交易列表查看交易情况,如果未找到订单,需要商户重新发起支付交易;如果订单显示成功支付,商户收银系统再次调用【查询订单API】查询实际支付结果;

(2)用户微信端弹出支付失败提示,例如:余额不足,信用卡失效。需要重新发起支付;

(3)当交易超时或支付交易失败,商户收银系统必须调用【撤销订单API】,撤销此交易。

(4)由于银行系统异常、用户余额不足、不支持用户卡种等原因使当前支付交易失败,商户收银系统应该把错误提示明确展示给收银员。

(5)根据返回的错误码,判断是否需要撤销交易,具体详见API返回错误码列表

下面是微信支付的协议规则:

微信支付也要使用到签名(加密)

关于签名:

第一步生成签名:

(1)生成签名,在生成签名之前,与支付宝的一样,也是要将订单的信心以URL键值对的格式即key1=value1&key2=value2…)拼接成字符串stringA。对于该键值对要进行排序,按照(参数名ASII码排序,就是第一个字母排序,如果首字母相同,就对第二个字母进行排序,依次排序....)

(2)不对空的键值对进行签名,在准备订单信息的时候如出现空值,则该空值对应的键不被签名

(3)参数有大小写之分,并且键 sign 不参与签名

(4)如果订单中除了微信给出的字段,又增添若干字段,

第二步得到临时sign:

在stringA最后拼接上key得到stringSignTemp字符串,并对stringSignTemp进行MD5运算,再将得到的字符串所有字符转换为大写,得到sign值signValue。

key设置路径:微信商户平台(pay.weixin.qq.com)-->账户设置-->API安全-->密钥设置

样例参照官方文档:参照  如何使用在线签名验证的操作:参照

在官方文档中说明 “生成个随机数算法”  有这样一句话:字段nonce_str,主要保证签名不可预测。我觉得这与支付宝中的:时间戳、随机串都是要实现一样的功能,增加签名的复杂性与唯一性。微信这个nonce_str就相当于支付宝的Order.h中声明的时间戳属性字段。

在微信中凡是涉及资金的接口都需要获取商户的证书。如:资金回滚、退款、撤销接口。

关于商户证书:参考 就支付宝签名要用到的公钥与私钥与微信的签名机制比较,微信是采用的是证书机制,首先在商户服务器上部署微信根证书,私钥包含在证书里面,要想对订单信息进行签名,就需要对证书进行验证,由于根证书放在商户的有访问权限控制的非虚拟目录中(防止非法下载,防止病毒木马,他人非法盗取),所以不论是请求还是回调都要使用https安全的传输协议。

注意事项:微信官方文档建议商户使用较高版本的微信SSL、建议在调用支付API时,使用IPV4解析 参考

微信关于支付这一块提供的官方接口主要有:提交刷卡支付API查询订单API撤销订单API申请退款API查询退款下载对账单API转换短连接API授权码查询API

要使用微信支付前,必须要仔细阅读官方文档,要做到对上面的API的各种参数了解,明白怎么使用,也要注意文档中的注意事项。

有可能官方的SDK会修改,文档会变动,所以还是要及时关注官方的文档去学习。在我的下一篇博客中我会去重点写一下使用微信支付的SDK环境搭建,以及如何从微信的官方demo中抽出有用的组件到我们的工程中。关于主要的参数本篇不再赘述。

下面仅仅是模拟微信支付的实现,因为微信支付是在微信这个App中不存在调用网页支付。所以本文不能看到支付的界面。

首先建立一个新的工程作为我们的工程,工程名字为weixin_study

使用故事版为其关联一个支付的按钮,按钮名字为:weixinPay_Button

(1)注册微信的开放平台申请Appid 注册

这里注册的测试 Appid 是 “wx920fde9f97d60569”

在工程->info->URL Types 中作如下配置

(2)从官网得到的demo中取出名字为 “SDKExport” 的文件夹拖入我们的工程中

(2)在工程->Build Phases->Link Binary With Libraries 中引入下面组件

致此支付的环境已经配置好了,下面实现支付的功能

我们模拟的订单信息是:

{
"status": 200,
"message": "成功!",
"data": {
"payId": 14187,
"orderId": 16257,
"orderNo": "2016011215151844904636",
"amount": 300,
"name": "【内饰清洗】微利天弘汽车美容装饰",
"description": "微利天弘汽车美容装饰 内饰清洗服务",
"notifyUrl": "http://app.cheguchina.com/wash/weixinpay/mobilenotify",
"payType": 30,
"ip": "218.28.20.138",
"prepay": {
"appId": "wx920fde9f97d60569",
"partnerId": "1220277201",
"prepayId": "920103900011129271ad0485b2fa00",
"package": "Sign=WXPay",
"timeStamp": "1452582979",
"nonce": "xl27JbHJSLcs52tv",
"sign": "1f3da1f83f60cbda0f715e6b0b9a0241b790d4"
}
}
}

在Appdeleagate.m中注册商户App,,并导入相关头文件

下面是主要的关键代码:

#import "AppDelegate.h"
#import "WXApi.h"
#import "WXApiObject.h" @interface AppDelegate ()<WXApiDelegate> @end @implementation AppDelegate - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { //1.AppID : 微信开发者平台注册后给的一个应用标示
//2.description : 没什么用只是给开发者一种解释的作用
[WXApi registerApp:@"wx920fde9f97d60569" withDescription:@"demo 2.0"]; return YES;
} -(void)onResp:(BaseResp*)resp{ if ([resp isKindOfClass:[PayResp class]]){
PayResp *response=(PayResp*)resp; //通知中心传值
[[NSNotificationCenter defaultCenter] postNotificationName:@"weixinzhifu" object:response]; }
}

Appdeleagate.m

#import "ViewController.h"
#import "WXApi.h"
#import "WXApiObject.h"
@interface ViewController ()<WXApiDelegate> @end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
}
- (IBAction)wxAtcion:(id)sender { //微信支付方法 PayReq *request = [[PayReq alloc] init]; //** 商家向财付通申请的商家id */
request.partnerId = @"";
//预支付订单 (包含商品的信息 如价格 : 标题 , 描述 等信息)
request.prepayId= @"8201038000160122a3971f85dc6f20d";
//商家根据财付通文档填写的数据和签名 类似一种标示
request.package = @"Sign=WXPay";
//随机串
request.nonceStr= @"5xv1tGmFhe1cq5IZ";
//时间戳 (从1970年到现在的秒数)
request.timeStamp = ;
//商家根据微信开放平台文档对数据做的签名 (对数据的一种加密形势)
request.sign= @"0ac8963b331100acf5461e4d1ab726f7877912e"; //调用微信支付的方法
[WXApi sendReq:request]; } - (void)viewWillAppear:(BOOL)animated {
[super viewWillAppear:animated]; //接受通知
[[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(weiXinPayBack:) name:@"weixinzhifu" object:nil];
} //通知中心触发的方法
- (void)weiXinPayBack:(NSNotification *)notification { PayResp *response=notification.object; switch(response.errCode){
case WXSuccess:
//服务器端查询支付通知或查询API返回的结果再提示成功
NSLog(@"支付成功");
break;
default:
NSLog(@"支付失败,retcode=%d",response.errCode);
break;
}
} - (void)viewWillDisappear:(BOOL)animated {
[super viewWillDisappear:animated];
[[NSNotificationCenter debugDescription] removeObserver:self forKeyPath:@"weixinzhifu"];
} - (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
} @end

ViewController.m

这里已经能够完成实现支付了,如果数据是真正从服务器获得的数据就可以查看到支付结果了。

最新文章

  1. ueditor的配置和使用
  2. .NET LINQ 投影运算
  3. leetcode 137
  4. Android——程序移植 相关知识总结贴
  5. 【BZOJ】1082: [SCOI2005]栅栏(二分+dfs)
  6. LeetCode Meeting Rooms II
  7. noip2011普及组——统计单词数
  8. Delphi 记事本 TMemo(5篇)
  9. aps.net 用ajax 读取服务端值
  10. java IO 实例分析
  11. M0、M1、M2、M3都是用来反映货币供应量的重要指标
  12. 无法识别的配置节 system.serviceModel
  13. Android Studio中.9.png文件出错问题
  14. Bootstrap下拉菜单的使用(附源码文件)--Bootstrap
  15. WebStrom中实现Vue项目的快速启动
  16. Java11新特性!
  17. MySQL 压缩文件安装遇到的问题及解决方案
  18. Java原生API访问MongoDB
  19. Hibernate----面试题
  20. (四)Spring 对DAO 的支持

热门文章

  1. Java中HashMap遍历的两种方法(转)
  2. Web知识点收集
  3. mediawiki 管理员/行政员设置
  4. vim 宏的使用
  5. 算法NB三人组
  6. Java解析Property文件
  7. Java ClassLoader详解(转载)
  8. WPF3.5 使用BINDINGGROUP进行实体类和集合验证
  9. MVC——分页
  10. 程序猿的量化交易之路(27)--Cointrader之PriceData价格数据(14)