欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

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

程序员文章站 2023-11-22 10:32:34
前段时间做了网页版微信支付,遇到很多问题,不过最终还是解决了,现在在这里记录下开发流程以及说明,给其他人一些参考。 一、准备工作 首先肯定得先要开通微信支付功能,之前开...

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

一、准备工作

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

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

1、开发目录配置

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

微信授权目录需要精确到二级或三级目录,事例:假如发起支付的链接是 http://www.hxfspace.net/weixin/wexinpay/wexinpaychoose  那么配置的目录应该是http://www.hxfspace.net/weixin/wexinpay/ 其中 http://www. hxfspace.net是域名weixin是虚拟目录 wexinpay也就是controller 相关的支付请求都在wexinpay中的action里面。     解析微信支付的实现方法(.NET版)           

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

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

3、相关参数准备

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

解析微信支付的实现方法(.NET版)其中package和paysign的生成需要开发者密钥appsecret(应用密钥)、微信商户号、微信支付密钥,这些参数的获取和设置可以看这篇文章

二、开发流程

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

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

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