我的编程空间,编程开发者的网络收藏夹
学习永远不晚

IOS客户端接入微信支付

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

IOS客户端接入微信支付

实际上,从代码的角度,调起支付APP就是把一些关键的参数通过一定方式打包成为一个订单,然后发送到支付平台的服务器。所以,只要搞清楚了参数设置,搞清楚了每个支付平台的SDK里面一些关键API的使用,基本上就可以很简单的支持支付。

今天记录一下客户端里面,如何支持微信支付。首先。我们要仔细阅读一下微信SDK的开发文档,了解一下整个支付的大概流程。

然后根据提示,把相应的SDK下载下来,所谓的SDK,也就是一个链接库和两个头文件,很简单。

下载完毕,需要把SDK导入到工程里面,并且配置一下工程。因为开发者文档已经有详细描述,这里就不再复述。

从文档看到,调起微信支付其实最核心的是一下这么一段


<code class="hljs" vbscript="">PayReq *request = [[[PayReq alloc] init] autorelease];
request.partnerId = @10000100;
request.prepayId= @1101000000140415649af9fc314aa427;
request.package = @Sign=WXPay;
request.nonceStr= @a462b76e7436e98e0ed6e13c64b4fd1c;
request.timeStamp= @1397527777;
request.sign= @582282D72DD2B03AD892830965F428CB16E7A256;
[WXApi sendReq:request];</code>

这里的范例是一段hardcode,真正使用的时候,参数都需要自行传入。

为了搞清楚如何使用API,我们可以下载Sample代码。不过,这个sample代码应该是微信的实习生写的,而且应该是一个对于C++比较熟悉,对于ObjectC比较陌生的实习生。。。代码风格可以看出很多东西哈。。所以这个sample读起来总觉得有点奇怪。当然,写出这个demo也是需要不错的水平,因为这个sample不仅仅是一些API的调用,还包括了一些算法的实现,MD5之类的。 看懂了sample之后,一般可以自己重构一下,成为自己APP里面的一个Manager类。

我是在2015 5 23下载的微信Sampel代码,里面包括有:

ApiXml.h

ApiXml.m

WXUtil.h

WXUtil.m

payRequestHandler.h

payRequestHandler.m

如果比较看重命名规范的OC程序猿,就会觉得这个payRequestHandler类非常别扭,不符合camel命名规则,而且handler这个词更偏向于c++风格。我就以这个类为原型,重构了一下,并改装成一个传参的方法,供自己的APP调用。APP里面卖商品,一般就是商品名字,价格两个关键参数。所以这个重构的方法也只是提供这两个参数的接口。 ApiXml.h && ApiXml.m && WXUtil.h && WXUtil.m不变


<code class="hljs" objectivec="">//
// WechatPayManager.h
//
// Created by HuangCharlie on 5/24/15.
//
//
#import <foundation foundation.h="">
#import WXUtil.h
#import ApiXml.h
#import WXApi.h
// 账号帐户资料
// 更改商户把相关参数后可测试
#define APP_ID     @wx@@@@@@@@@@@@@@@@    //APPID
#define APP_SECRET   @             //appsecret,看起来好像没用
//商户号,填写商户对应参数
#define MCH_ID     @@@@@@@@@@@
//商户API密钥,填写相应参数
#define PARTNER_ID   @12345678901234567890123456789012
//支付结果回调页面
#define NOTIFY_URL   @http://wxpay.weixin.qq.com/pub_v2/pay/notify.v2.php
//获取服务器端支付数据地址(商户自定义)(在小吉这里,签名算法直接放在APP端,故不需要自定义)
#define SP_URL     @http://wxpay.weixin.qq.com/pub_v2/app/app_pay.php
@interface WechatPayManager : NSObject
{
}
//预支付网关url地址
@property (nonatomic,strong) NSString* payUrl;
//debug信息
@property (nonatomic,strong) NSMutableString *debugInfo;
@property (nonatomic,assign) NSInteger lastErrCode;//返回的错误码
//商户关键信息
@property (nonatomic,strong) NSString *appId,*mchId,*spKey;
//初始化函数
-(id)initWithAppID:(NSString*)appID
       mchID:(NSString*)mchID
       spKey:(NSString*)key;
//获取当前的debug信息
-(NSString *) getDebugInfo;
//获取预支付订单信息(核心是一个prepayID)
- (NSMutableDictionary*)getPrepayWithOrderName:(NSString*)name
                     price:(NSString*)price
                    device:(NSString*)device;
@end
</foundation></code>
<code class="hljs" objectivec="">//
// WechatPayManager.m
//
// Created by HuangCharlie on 5/24/15.
//
//
#import WechatPayManager.h
@implementation WechatPayManager
//初始化函数
-(id)initWithAppID:(NSString*)appID mchID:(NSString*)mchID spKey:(NSString*)key
{
  self = [super init];
  if(self)
  {
    //初始化私有参数,主要是一些和商户有关的参数
    self.payUrl  = @https://api.mch.weixin.qq.com/pay/unifiedorder;
    if (self.debugInfo == nil){
      self.debugInfo = [NSMutableString string];
    }
    [self.debugInfo setString:@];
    self.appId = appID;//微信分配给商户的appID
    self.mchId = mchID;//
    self.spKey = key;//商户的密钥
  }
  return self;
}
//获取debug信息
-(NSString*) getDebugInfo
{
  NSString *res = [NSString stringWithString:self.debugInfo];
  [self.debugInfo setString:@];
  return res;
}
//创建package签名
-(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=%@, self.spKey];
  //得到MD5 sign签名
  NSString *md5Sign =[WXUtil md5:contentString];
  //输出Debug Info
  [self.debugInfo appendFormat:@MD5签名字符串:
%@
,contentString];
  return md5Sign;
}
//获取package带参数的签名包
-(NSString *)genPackage:(NSMutableDictionary*)packageParams
{
  NSString *sign;
  NSMutableString *reqPars=[NSMutableString string];
  //生成签名
  sign    = [self createMd5Sign:packageParams];
  //生成xml的package
  NSArray *keys = [packageParams allKeys];
  [reqPars appendString:@<xml>
];
  for (NSString *categoryId in keys) {
    [reqPars appendFormat:@<%@>%@<!--%@-->
, categoryId, [packageParams objectForKey:categoryId],categoryId];
  }
  [reqPars appendFormat:@<sign>%@</sign>
</xml>, sign];
  return [NSString stringWithString:reqPars];
}
//提交预支付
-(NSString *)sendPrepay:(NSMutableDictionary *)prePayParams
{
  NSString *prepayid = nil;
  //获取提交支付
  NSString *send   = [self genPackage:prePayParams];
  //输出Debug Info
  [self.debugInfo appendFormat:@API链接:%@
, self.payUrl];
  [self.debugInfo appendFormat:@发送的xml:%@
, send];
  //发送请求post xml数据
  NSData *res = [WXUtil httpSend:self.payUrl method:@POST data:send];
  //输出Debug Info
  [self.debugInfo appendFormat:@服务器返回:
%@
,[[NSString alloc] initWithData:res encoding:NSUTF8StringEncoding]];
  XMLHelper *xml = [[XMLHelper alloc] autorelease];
  //开始解析
  [xml startParse:res];
  NSMutableDictionary *resParams = [xml getDict];
  //判断返回
  NSString *return_code  = [resParams objectForKey:@return_code];
  NSString *result_code  = [resParams objectForKey:@result_code];
  if ( [return_code isEqualToString:@SUCCESS] )
  {
    //生成返回数据的签名
    NSString *sign   = [self createMd5Sign:resParams ];
    NSString *send_sign =[resParams objectForKey:@sign] ;
    //验证签名正确性
    if( [sign isEqualToString:send_sign]){
      if( [result_code isEqualToString:@SUCCESS]) {
        //验证业务处理状态
        prepayid  = [resParams objectForKey:@prepay_id];
        return_code = 0;
        [self.debugInfo appendFormat:@获取预支付交易标示成功!
];
      }
    }else{
      self.lastErrCode = 1;
      [self.debugInfo appendFormat:@gen_sign=%@
  _sign=%@
,sign,send_sign];
      [self.debugInfo appendFormat:@服务器返回签名验证错误!!!
];
    }
  }else{
    self.lastErrCode = 2;
    [self.debugInfo appendFormat:@接口返回错误!!!
];
  }
  return prepayid;
}
- (NSMutableDictionary*)getPrepayWithOrderName:(NSString*)name
                     price:(NSString*)price
                    device:(NSString*)device
{
  //订单,展示给用户
  NSString* orderName = name;
  //订单金额,单位(分)
  NSString* orderPrice = price;//以分为单位的整数
  //支付设备号或门店号
  NSString* orderDevice = device;
  //支付类型,固定为APP
  NSString* orderType = @APP;
  //发器支付的机器ip,暂时没有发现其作用
  NSString* orderIP = @196.168.1.1;
  //随机数串
  srand( (unsigned)time(0) );
  NSString *noncestr = [NSString stringWithFormat:@%d, rand()];
  NSString *orderNO  = [NSString stringWithFormat:@%ld,time(0)];
  //================================
  //预付单参数订单设置
  //================================
  NSMutableDictionary *packageParams = [NSMutableDictionary dictionary];
  [packageParams setObject: self.appId forKey:@appid];    //开放平台appid
  [packageParams setObject: self.mchId forKey:@mch_id];   //商户号
  [packageParams setObject: orderDevice forKey:@device_info]; //支付设备号或门店号
  [packageParams setObject: noncestr   forKey:@nonce_str];  //随机串
  [packageParams setObject: orderType  forKey:@trade_type]; //支付类型,固定为APP
  [packageParams setObject: orderName  forKey:@body];    //订单描述,展示给用户
  [packageParams setObject: NOTIFY_URL forKey:@notify_url]; //支付结果异步通知
  [packageParams setObject: orderNO   forKey:@out_trade_no];//商户订单号
  [packageParams setObject: orderIP   forKey:@spbill_create_ip];//发器支付的机器ip
  [packageParams setObject: orderPrice  forKey:@total_fee];    //订单金额,单位为分
  //获取prepayId(预支付交易会话标识)
  NSString *prePayid;
  prePayid = [self sendPrepay:packageParams];
  if(prePayid == nil)
  {
    [self.debugInfo appendFormat:@获取prepayid失败!
];
    return nil;
  }
  //获取到prepayid后进行第二次签名
  NSString  *package, *time_stamp, *nonce_str;
  //设置支付参数
  time_t now;
  time(&now);
  time_stamp = [NSString stringWithFormat:@%ld, now];
  nonce_str = [WXUtil md5:time_stamp];
  //重新按提交格式组包,微信客户端暂只支持package=Sign=WXPay格式,须考虑升级后支持携带package具体参数的情况
  //package    = [NSString stringWithFormat:@Sign=%@,package];
  package     = @Sign=WXPay;
  //第二次签名参数列表
  NSMutableDictionary *signParams = [NSMutableDictionary dictionary];
  [signParams setObject: self.appId forKey:@appid];
  [signParams setObject: self.mchId forKey:@partnerid];
  [signParams setObject: nonce_str  forKey:@noncestr];
  [signParams setObject: package   forKey:@package];
  [signParams setObject: time_stamp  forKey:@timestamp];
  [signParams setObject: prePayid   forKey:@prepayid];
  //生成签名
  NSString *sign = [self createMd5Sign:signParams];
  //添加签名
  [signParams setObject: sign     forKey:@sign];
  [self.debugInfo appendFormat:@第二步签名成功,sign=%@
,sign];
  //返回参数列表
  return signParams;
}
@end</code>

然后,在需要调用微信支付的Controller里面,新建一个方法。在合适的地方调用。这个方法里面利用WechatPayManager这个类进行了初始化和参数封装,然后把上述的核心代码(PayReq那一段)


<code class="hljs" objectivec="">- (void)wxPayWithOrderName:(NSString*)name price:(NSString*)price
{
  //创建支付签名对象 && 初始化支付签名对象
  WechatPayManager* wxpayManager = [[[WechatPayManager alloc]initWithAppID:APP_ID mchID:MCH_ID spKey:PARTNER_ID] autorelease];
 
  //获取到实际调起微信支付的参数后,在app端调起支付
  //生成预支付订单,实际上就是把关键参数进行第一次加密。
  NSString* device = [[UserManager defaultManager]userId];
  NSMutableDictionary *dict = [wxpayManager getPrepayWithOrderName:name
                                price:price
                             device:device];
  if(dict == nil){
    //错误提示
    NSString *debug = [wxpayManager getDebugInfo];
    return;
  }

  NSMutableString *stamp = [dict objectForKey:@timestamp];
 
  //调起微信支付
  PayReq* req       = [[[PayReq alloc] init]autorelease];
  req.openID       = [dict objectForKey:@appid];
  req.partnerId     = [dict objectForKey:@partnerid];
  req.prepayId      = [dict objectForKey:@prepayid];
  req.nonceStr      = [dict objectForKey:@noncestr];
  req.timeStamp     = stamp.intValue;
  req.package      = [dict objectForKey:@package];
  req.sign        = [dict objectForKey:@sign];
 
//    BOOL flag = [WXApi sendReq:req];
  BOOL flag = [WXApi safeSendReq:req];
}</code>

再者,支付完成了需要调用一个delegate,这个delegate方便个性化显示支付结果。一般直接把这两个delegate放在AppDelegate就好了。因为有一些其他内容也是需要在AppDelegate里面实现,省的分开找不到。


<code class="hljs" objectivec="">-(void) onResp:(BaseResp*)resp
{ 
  //启动微信支付的response
  NSString *strMsg = [NSString stringWithFormat:@errcode:%d, resp.errCode];
  if([resp isKindOfClass:[PayResp class]]){
    //支付返回结果,实际支付结果需要去微信服务器端查询
    switch (resp.errCode) {
      case 0:
        strMsg = @支付结果:成功!;
        break;
      case -1:
        strMsg = @支付结果:失败!;
        break;
      case -2:
        strMsg = @用户已经退出支付!;
        break;
      default:
        strMsg = [NSString stringWithFormat:@支付结果:失败!retcode = %d, retstr = %@, resp.errCode,resp.errStr];
        break;
    }
  }
}</code>

注意事项:

1)如果APP里面已经使用了ShareSDK,就有一些地方要注意。不要再重复导入微信的SDK,因为shareSDK里面的extend已经包括了微信的SDK。

2)微信本身是鼓励客户APP把签名算法放到服务器上面,这样信息就不容易被破解。但是如果客户APP本身没有服务器端,或者认为不需要放到服务器端,也可以直接把签名(加密)的部分直接放在APP端。Sample代码的注释有点乱,多次提到服务器云云,但是其实可以不这么做。

3)微信的price单位是分。注意下即可。

4)暂时想不到,以后想到了再记录。。

以上是本文给大家叙述的IOS客户端接入微信支付的全部内容,希望大家喜欢。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

IOS客户端接入微信支付

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

IOS客户端接入微信支付

实际上,从代码的角度,调起支付APP就是把一些关键的参数通过一定方式打包成为一个订单,然后发送到支付平台的服务器。所以,只要搞清楚了参数设置,搞清楚了每个支付平台的SDK里面一些关键API的使用,基本上就可以很简单的支持支付。 今天记录一下
2022-05-20

〔支付接入〕微信的 h5 支付和 jsapi 支付

✨ 目录 🎈 申请商户号🎈 申请商户证书🎈 设置APIv3密钥🎈 下载 SDK 开发包🎈 下载平台证书🎈 关联 AppID 账
2023-08-16

Java后端如何对接微信支付

本篇内容主要讲解“Java后端如何对接微信支付”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java后端如何对接微信支付”吧!首先我们要明确目标,我们点击微信支付官网,我们主要聚焦于这三种支付方
2023-06-22

手把手教你接入微信支付

随着微信小程序的发展,越来越多的移动端应用选择了微信产品作为媒介。无论是公众号开发还是小程序开发,微信支付永远都是绕不开的话题。本文我们只学习如何在公众号、小程序中接入微信支付。

Android App支付系列(一):微信支付接入详细指南(附官方支付demo)

写在前面 一家移动互联网公司,说到底,要盈利总是需要付费用户的,自己开发支付系统显然是不明智的,国内已经有多家成熟的移动支付提供商,腾讯就是其中之一。梳理了下微信支付的接入,今天给大家分享下腾讯旗下的微信支付SDK的接入流程。 接入流程 1
2022-06-06

微信小程序接入微信支付实现过程详解

这篇文章主要介绍了微信小程序接入微信支付实现过程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
2022-12-26

Android 支付宝支付、微信支付、银联支付 整合第三方支付接入方法(后台订单支付API设计)

客户端获取后台支付API请求参数的设计参数样例:{ data: { method: 1, platform: 1, version:"1.0", relate_orders:"B201602031023,B2016020310231", o
2022-06-06

微信支付、支付宝接入指引,看完立刻就可以上手!

前段时间阿粉在公司接手了一个支付项目,这个项目接入了微信、支付宝。这个项目开发下来,阿粉可是完完整整体验了一下微信、支付宝开发流程,也踩了一些坑。

企业微信活跃用户达1.8亿!微信客服正式接入

1月11日消息,今日,企业微信公众号公布了企业微信的最新战绩,企业微信上的真实企业与组织数超1000万,活跃用户数达1.8亿,连接微信活跃用户数超过5亿。

Java接入微信支付超级详细教程——从入门到精通

源码下载 源码获取:点击获取源码 本文介绍了“二维码付款”的代码。其他微信支付方式的代码都在源码中。 一、准备开发所需的账号以及配置信息 解释:想要接入微信支付我们需要两个玩意 ,一个是公众号/小程序/企业微信(开发用的),一个是微信支付商
2023-08-17

腾讯接入!微信支持数字人民币支付:两步即可开通

据国内媒体报道,近日,数字人民币App升级更新,微众银行(微信支付)数字人民币钱包随之上线。经前期多阶段可控试点后,腾讯开始为用户提供数字人民币服务。

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录