侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

解析微信支付的实现方法(.NET版)

微信(小程序)  /  管理员 发布于 7年前   139

前段时间做了网页版微信支付,遇到很多问题,不过最终还是解决了,现在在这里记录下开发流程以及说明,给其他人一些参考。

一、准备工作

首先肯定得先要开通微信支付功能,之前开通微信支付需要三万的押金的,现在不需要了,所以就做了这个功能。

要进行微信支付开发,需要在公众号后台和微信商户后台进行相关的设置。

1、开发目录配置

微信支付需要在公众号后台(微信支付=》开发配置)进行配置支付授权目录。这里授权目录需要是线上地址,也就是可以通过互联网访问到的地址,微信支付系统需要能够通过互联网访问到你的地址。

微信授权目录需要精确到二级或三级目录,事例:假如发起支付的链接是 http://www.hxfspace.net/weixin/WeXinPay/WeXinPayChoose  那么配置的目录应该是http://www.hxfspace.net/weixin/WeXinPay/ 其中 http://www. hxfspace.net是域名weixin是虚拟目录 WeXinPay也就是Controller 相关的支付请求都在WeXinPay中的action里面。                

 2、OAuth2.0网页授权域名设置

微信支付的时候会对支付请求进行回调来获取授权代码(code),所以需要在这里设置授权域名。当然这里域名是要和支付授权目录中的域名是同一个。这个不要忘记设置了我当时就是忘记设置然后找半天原因,哭死。

3、相关参数准备

调用微信支付需要通过脚本向微信支付系统发起支付请求,参数说明见微信官网支付平台https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_7&index=6 

其中package和paySign的生成需要开发者密钥AppSecret(应用密钥)、微信商户号、微信支付密钥,这些参数的获取和设置可以看这篇文章softjc/346871.html

二、开发流程

废话不多说直接说整理之后的流程:

1、通过微信授权回调来获取授权code

2、通过授权code来换取网页授权access_token 和openid

3、调用统一下单接口获取预支付prepayId

4、组建jsapi微信支付请求参数,发起支付

5、接收微信支付回调进行后续操作

三、具体开发(上代码)

微信支付只能在线上环境中进行,调式很不方便,所在在刚开始开发的时候最好在每个关键位置记录好日志。

1、通过微信授权回调来获取授权code

首先把发起支付地址以及相关参数传给微信支付接口,微信支付接收验证成功之后,会重新请求你的支付地址并带上授权code。

比如我这里
  

 //判断是否网页授权,获取授权code,没有代表没有授权,构造网页授权获取code,并重新请求      if (string.IsNullOrEmpty(Request.QueryString["code"]))      {        string redirectUrl = _weChatPaySerivce.GetAuthorizeUrl(account.AppId, account.RedquestUrl,          "STATE" + "#wechat_redirect", "snsapi_base");        return Redirect(redirectUrl);      }

拼接微信网页授权Url方法

public string GetAuthorizeUrl(string appId, string redirectUrl, string state, string scope)    {      string url = string.Format("https://open.weixin.qq.com/connect/oauth2/authorize?appid={0}&redirect_uri={1}&response_type=code&scope={2}&state={3}",          appId, HttpUtility.UrlEncode(redirectUrl), scope, state);      /* 这一步发送之后,客户会得到授权页面,无论同意或拒绝,都会返回redirectUrl页面。       * 如果用户同意授权,页面将跳转至 redirect_uri/?code=CODE&state=STATE。这里的code用于换取access_token(和通用接口的access_token不通用)       * 若用户禁止授权,则重定向后不会带上code参数,仅会带上state参数redirect_uri?state=STATE       */      AppLog.Write("获取到授权url:", AppLog.LogMessageType.Debug);       return url;    }

2、通过授权code来换取网页授权access_token 和openid

从第一步中获取到授权code之后,组合网页授权请求url,来获取access_token 和openid

 public Tuple<string, string> GetOpenidAndAccessTokenFromCode(string appId, string code, string appSecret)    {      Tuple<string, string> tuple = null;      try      {        string url = string.Format("https://api.weixin.qq.com/sns/oauth2/access_token?appid={0}&secret={1}&code={2}&grant_type=authorization_code", appId, appSecret, code);        string result = WeChatPayHelper.Get(url);        AppLog.Write("微信支付-获取openid和access_token 请求Url:" + url + "result:" + result, AppLog.LogMessageType.Debug);        if (!string.IsNullOrEmpty(result))        {          var jd=Newtonsoft.Json.JsonConvert.DeserializeObject<Dictionary<string, string>>(result);          tuple = new Tuple<string, string>(jd["openid"],jd["access_token"]);          AppLog.Write("微信支付-获取openid和access_token成功", AppLog.LogMessageType.Debug);        }      }      catch (Exception ex)      {        AppLog.Write("微信支付:获取openid和access_tokenu异常", AppLog.LogMessageType.Debug,ex);      }      return tuple;    }

3、调用统一下单接口获取预支付prepayId

 这里RequestHandler是用的网上别人封装好的dll,帮你封装好了签名的生成以及一些验证请求。dll可以在这他们官网下载http://weixin.senparc.com/

//创建支付应答对象      RequestHandler packageReqHandler = new RequestHandler(null);      //初始化      packageReqHandler.Init();      //时间戳      string timeStamp = TenPayUtil.GetTimestamp();      //随机字符串      string nonceStr = TenPayUtil.GetNoncestr();      //设置package订单参数 生成prepayId预支付Id      packageReqHandler.SetParameter("appid", account.AppId);     //公众账号ID      packageReqHandler.SetParameter("mch_id", account.PartnertId);     //商户号      packageReqHandler.SetParameter("nonce_str", nonceStr);          //随机字符串      packageReqHandler.SetParameter("body", account.Body);      packageReqHandler.SetParameter("out_trade_no", account.OrderSerialId);    //商家订单号      packageReqHandler.SetParameter("total_fee", account.TotalAmount);          //商品金额,以分为单位(money * 100).ToString()      packageReqHandler.SetParameter("spbill_create_ip", account.RequestIp);  //用户的公网ip,不是商户服务器IP      packageReqHandler.SetParameter("notify_url", account.NotifyUrl);      //接收财付通通知的URL      packageReqHandler.SetParameter("trade_type", "JSAPI");//交易类型      packageReqHandler.SetParameter("openid", account.OpenId);//用户的openId      string sign = packageReqHandler.CreateMd5Sign("key", account.PaySignKey);      packageReqHandler.SetParameter("sign", sign);//签名      string prepayId = string.Empty;      try      {        string data = packageReqHandler.ParseXML();        var result = TenPayV3.Unifiedorder(data);        MailHelp.SendMail("调用统一下单接口,下单结果:--"+result+"请求参数:"+data);        var res = XDocument.Parse(result);        prepayId = res.Element("xml").Element("prepay_id").Value;        AppLog.Write("调用统一下单接口获取预支付prepayId成功", AppLog.LogMessageType.Debug);      }      catch (Exception ex)      {        AppLog.Write("获取到openid和access_tokenu异常", AppLog.LogMessageType.Debug, ex);        MailHelp.SendMail("调用统一下单接口获取预支付prepayid异常:", ex);        return null;      }

4、组建jsapi微信支付请求参数,发起支付

我这里是首先组装好微信支付所需要的参数,然后再创建调用js脚本    

//生成JsAPI支付参数      RequestHandler paySignReqHandler = new RequestHandler(null);      paySignReqHandler.SetParameter("appId", account.AppId);      paySignReqHandler.SetParameter("timeStamp", timeStamp);      paySignReqHandler.SetParameter("nonceStr", nonceStr);      paySignReqHandler.SetParameter("package", string.Format("prepay_id={0}", prepayId));      paySignReqHandler.SetParameter("signType", "MD5");      string paySign = paySignReqHandler.CreateMd5Sign("key", account.PaySignKey);      WeChatJsPayRequestModel resultModel = new WeChatJsPayRequestModel      {        AppId = account.AppId,        NonceStr = nonceStr,        TimeStamp = timeStamp,        Package = string.Format("prepay_id={0}", prepayId),        PaySign = paySign,        SignType = "MD5"      };

创建调用脚本

private string CreateWeixinJs(WeChatJsPayRequestModel model)    {      string js = @"<script type='text/javascript'>    callpay();    function jsApiCall(){     WeixinJSBridge.invoke(      'getBrandWCPayRequest', {        requestParam      },      function (res) {        if(res.err_msg == 'get_brand_wcpay_request:ok' ){window.location.href = 'successUrl';        }else{window.location.href = 'failUrl';        }      }     );     }   function callpay()    {      if (typeof WeixinJSBridge == 'undefined'){        if( document.addEventListener ){          document.addEventListener('WeixinJSBridgeReady', jsApiCall, false);        }else if (document.attachEvent){          document.attachEvent('WeixinJSBridgeReady', jsApiCall);           document.attachEvent('onWeixinJSBridgeReady', jsApiCall);        }      }else{        jsApiCall();      }    }</script>";      string requestParam = string.Format(@"'appId': '{0}','timeStamp': '{1}','nonceStr': '{2}','package': '{3}','signType': '{4}','paySign': '{5}'",        model.AppId, model.TimeStamp, model.NonceStr, model.Package, model.SignType, model.PaySign);      js = js.Replace("requestParam", requestParam)        .Replace("successUrl", model.JumpUrl + "&result=1")        .Replace("failUrl", model.JumpUrl + "&result=0");      AppLog.Write("生成可执行脚本成功", AppLog.LogMessageType.Debug);      return js;    }

 5、接收微信支付回调进行后续操作

回调的时候首先需要验证签名是否正确,保证安全性,签名验证通过之后再进行后续的操作,订单状态、通知啥的。 

ResponseHandler resHandler = new ResponseHandler(System.Web.HttpContext.Current);      bool isSuccess = _weChatPaySerivce.ProcessNotify(resHandler);      if (isSuccess)      {        string result = @"<xml>      <return_code><![CDATA[SUCCESS]]></return_code>      <return_msg><![CDATA[支付成功]]></return_msg>     </xml>";        HttpContext.Response.Write(result);        HttpContext.Response.End();      }      return new EmptyResult();

这里有一点需要注意,就是微信支付回调的时候微信会通知八次,好像是这个数吧,所以你需要在第一次收到通知之后,把收到请求这个状态以xml的格式响应给微信支付接口。当然你不进行这个操作也是可以的,再回调的时候 每次去判断该订单是否已经回调成功,回调成功则不进行处理就可以了。

 原文链接:http://www.cnblogs.com/minesnil-forfaith/p/4976006.html

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


  • 上一条:
    ASP.NET微信公众号客服接口
    下一条:
    C#开发微信 二维码鼠标滑动 图像显示隐藏效果(推荐)
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 微信模板消息改版后发送规则记录(微信订阅消息参数值内容限制说明)(1个评论)
    • 微信支付v3对接所需工具及命令(0个评论)
    • 2023年9月1日起:微信小程序必须备案才能上线运营(0个评论)
    • 腾讯官方客服回应了:微信好友上限约10000个!(1个评论)
    • 2023年做微信小程序的老铁注意:新增收费项、微信小程序获取手机号也收费了(2个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2017-10
    • 2018-01
    • 2020-03
    • 2021-06
    • 2021-10
    • 2022-03
    • 2023-02
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-10
    • 2023-11
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客