首页 > PHP资讯 > 工具库 > .NET微信公众号开发(5.0微信支付)实例详解

.NET微信公众号开发(5.0微信支付)实例详解

工具库

一.前言

在开始做这个功能之前,我们要做的第一件事情就是思考,如何做这个微信支付,从哪里开始,从哪里入手,官方的sdk说明什么的,有没有什么官方的demo,还有就是老板给我的一些资料齐全不,那些要申请的接 口什么的都有没有。

经过自己的一些探索,在老板的催促下终于硬着头皮做完了这个,很坑很坑的微信支付,在此做一些总结,希望对你们有所帮助,本人能力有限,如果有什么说的不好,希望大家多多包涵。

二.开发前准备。

1.0微信支付官方开发者文档

2.0官方demo下载 我们用c#所以选择.net版本 不过这个官方的demo根本跑步起来

3.0官方demo运行起来解决方案

4.0微信支付官方.net版之坑你没商量

5.0开发前的微信公众平台的一些配置,请务必认真检查配置.

三.编码

做好了这些准备工作之后,我们知道微信支付有两种,1.原生态的,2.jsapi直接调用的,我项目中用到的是第二种

经过自己的一些业务逻辑处理,来到了我们的订单详情页面,现在需要去点击我们的支付按钮去支付,支付页面pay.aspx代码如下,

前台页面:

<script type="text/javascript">               //调用微信JS api 支付               function jsApiCall()               {                   WeixinJSBridge.invoke(                   'getBrandWCPayRequest',                   <%=wxJsApiParam%>,//josn串                    function (res)                    {                      if (res.err_msg == "get_brand_wcpay_request:ok")                       {                           var OrderId=$("#OrderId").val();                           var orderProductName=$("#orderProductName").val();                           var orderMoneySum=$("#orderMoneySum").val();                            window.location.href="http://www.baidu.aspx?OrderId="+OrderId+"&orderMoneySum="+orderMoneySum+"&orderProductName="+orderProductName;//支付成功后的跳转页面                        }else                        {                          WeixinJSBridge.call('closeWindow');                        }                                              }                    );               }               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》    




  您确认付款元...

需要注意的是:

微信JS API只能在微信内置浏览器中使用,其他浏览器调用无效。微信提供getBrandWCPayRequest接口供商户前端网页调用,调用之前微信会鉴定商户支付权限,若商户具有调起支付的权限,则将开始支付流程。 这里主要介绍支付前的接口调用规则,支付状态消息通知机制请参加下文。接口需要注意:所有传入参数都是字符串类型!

getBrandWCPayRequest参数如表6-5所示。

表6-6 getBrandWCPayRequest返回值

JS API的返回结果 get_brand_wcpay_request:ok 仅在用户成功完成支付时返回。由于前端交互复杂,get_brand_wcpay_request:cancel 或者 get_brand_wcpay_request:fail 可以统一处理为用户遇到错误或者主动放弃,不必细化区分。

pay.aspx后台页面代码:

               wxJsApiParam { ; ; }           Money { ;   OrderId { ;   orderMoneySum { ; ; }          orderProductName { ; ; }          Page_Load( (!=  JsApiPay( total_fee = Request[                        orderMoneySum = ParkName = Request[= ParkName+= Request[ (.IsNullOrWhiteSpace(total_fee)||total_fee==  WxPayException( +  + =.Parse((Convert.ToDouble(total_fee)*= (Convert.ToDouble(jsApiPay.total_fee)/=                         (Common.OpenId ==   WxPayException(=== jsApiPay.GetJsApiParameters(); +  + ex.InnerException.Message +  +  + ex.Message +

在这里需要我们注意的是:jsApiPay.openid = common.GetOpenId();

在支付的时候,我们需要首先获取用户的openId,然而获取用户openId的这个过程我们首先要进行Oauth2认证,在官方的demo中提供了JsApiPay.cs这个核心类库,里面已经有这个GetOpenidAndAccessToken()方法 。我这里是拿过来写成了自己的一个公共帮助类。Common.cs

    ///     /// 公共帮助类    ///     public class Common    {        private  HttpContext Context { get; set; }        public static string OpenId = "Openid";        public static string access_token = "access_token";        #region 构造函数        ///         /// 构造函数        ///         /// 
        ///         public string GetOpenId()        {                Common common = new Common(Context);                common.GetOpenidAndAccessToken();                return OpenId;        }         #endregion    }

 public class JsApiPay    {        ///         /// 保存页面对象,因为要在类的方法中使用Page的Request对象        ///         private Page page {get;set;}        ///         /// openid用于调用统一下单接口        ///         public string openid { get; set; }        ///         /// access_token用于获取收货地址js函数入口参数        ///         public string access_token { get; set; }        ///         /// 商品金额,用于统一下单        ///         public int total_fee { get; set; }        ///         /// 订单Id        ///         public string orderid { get; set; }        ///         /// 统一下单接口返回结果        ///         public WxPayData unifiedOrderResult { get; set; }         public JsApiPay(Page page)        {            this.page = page;        }        /**        *         * 网页授权获取用户基本信息的全部过程        * 详情请参看网页授权获取用户基本信息:
chapter=7_9        * @return string 共享收货地址js函数需要的参数,json格式可以直接做参数使用        */        public string GetEditAddressParameters()        {            string parameter = "";            try            {                string host = page.Request.Url.Host;                string path = page.Request.Path;                string queryString = page.Request.Url.Query;                //这个地方要注意,参与签名的是网页授权获取用户信息时微信后台回传的完整url                string url = "http://" + host + path + queryString;                //构造需要用SHA1算法加密的数据                WxPayData signData = new WxPayData();                signData.SetValue("appid",WxPayConfig.APPID);                signData.SetValue("url", url);                signData.SetValue("timestamp",WxPayApi.GenerateTimeStamp());                signData.SetValue("noncestr",WxPayApi.GenerateNonceStr());                signData.SetValue("accesstoken",access_token);                string param = signData.ToUrl();                Log.Debug(this.GetType().ToString(), "SHA1 encrypt param : " + param);                //SHA1加密                string addrSign = FormsAuthentication.HashPasswordForStoringInConfigFile(param, "SHA1");                Log.Debug(this.GetType().ToString(), "SHA1 encrypt result : " + addrSign);                //获取收货地址js函数入口参数                WxPayData afterData = new WxPayData();                afterData.SetValue("appId",WxPayConfig.APPID);                afterData.SetValue("scope","jsapi_address");                afterData.SetValue("signType","sha1");                afterData.SetValue("addrSign",addrSign);                afterData.SetValue("timeStamp",signData.GetValue("timestamp"));                afterData.SetValue("nonceStr",signData.GetValue("noncestr"));                //转为json格式                parameter = afterData.ToJson();                Log.Debug(this.GetType().ToString(), "Get EditAddressParam : " + parameter);            }            catch (Exception ex)            {                Log.Error(this.GetType().ToString(), ex.ToString());                throw new WxPayException(ex.ToString());            }            return parameter;        }    }

JsApiPay

微信支付协议接口数据类WxPayData.cs官方都有相应的代码.

 ///     /// 微信支付协议接口数据类,所有的API接口通信都依赖这个数据结构,    /// 在调用接口之前先填充各个字段的值,然后进行接口通信,    /// 这样设计的好处是可扩展性强,用户可随意对协议进行更改而不用重新设计数据结构,    /// 还可以随意组合出不同的协议数据包,不用为每个协议设计一个数据包结构    ///     public class WxPayData    {        public WxPayData()        {        }        //采用排序的Dictionary的好处是方便对数据包进行签名,不用再签名之前再做一次排序        private SortedDictionary m_values = new SortedDictionary();        /**        * 设置某个字段的值        * @param key 字段名         * @param value 字段值        */        public void SetValue(string key, object value)        {            m_values[key] = value;        }        /**        * 根据字段名获取某个字段的值        * @param key 字段名         * @return key对应的字段值        */        public object GetValue(string key)        {            object o = null;            m_values.TryGetValue(key, out o);            return o;        }        /**         * 判断某个字段是否已设置         * @param key 字段名         * @return 若字段key已被设置,则返回true,否则返回false         */        public bool IsSet(string key)        {            object o = null;            m_values.TryGetValue(key, out o);            if (null != o)                return true;            else                return false;        }        /**        * @将Dictionary转成xml        * @return 经转换得到的xml串        * @throws WxPayException        **/        public string ToXml()        {            //数据为空时不能转化为xml格式            if (0 == m_values.Count)            {                Log.Error(this.GetType().ToString(), "WxPayData数据为空!");                throw new WxPayException("WxPayData数据为空!");            }            string xml = "";            foreach (KeyValuePair pair in m_values)            {                //字段值不能为null,会影响后续流程                if (pair.Value == null)                {                    Log.Error(this.GetType().ToString(), "WxPayData内部含有值为null的字段!");                    throw new WxPayException("WxPayData内部含有值为null的字段!");                }                if (pair.Value.GetType() == typeof(int))                {                    xml += "<" + pair.Key + ">" + pair.Value + "";                }                else if (pair.Value.GetType() == typeof(string))                {                    xml += "<" + pair.Key + ">" + "";                }                else//除了string和int类型不能含有其他数据类型                {                    Log.Error(this.GetType().ToString(), "WxPayData字段数据类型错误!");                    throw new WxPayException("WxPayData字段数据类型错误!");                }            }            xml += "";            return xml;        }        /**        * @将xml转为WxPayData对象并返回对象内部的数据        * @param string 待转换的xml串        * @return 经转换得到的Dictionary        * @throws WxPayException        */        public SortedDictionary FromXml(string xml)        {            if (string.IsNullOrEmpty(xml))            {                Log.Error(this.GetType().ToString(), "将空的xml串转换为WxPayData不合法!");                throw new WxPayException("将空的xml串转换为WxPayData不合法!");            }            XmlDocument xmlDoc = new XmlDocument();            xmlDoc.LoadXml(xml);            XmlNode xmlNode = xmlDoc.FirstChild;//获取到根节点            XmlNodeList nodes = xmlNode.ChildNodes;            foreach (XmlNode xn in nodes)            {                XmlElement xe = (XmlElement)xn;                m_values[xe.Name] = xe.InnerText;//获取xml的键值对到WxPayData内部的数据中            }                        try            {                //2015-06-29 错误是没有签名                if(m_values["return_code"] != "SUCCESS")                {                    return m_values;                }                CheckSign();//验证签名,不通过会抛异常            }            catch(WxPayException ex)            {                throw new WxPayException(ex.Message);            }            return m_values;        }        /**        * @Dictionary格式转化成url参数格式        * @ return url格式串, 该串不包含sign字段值        */        public string ToUrl()        {            string buff = "";            foreach (KeyValuePair pair in m_values)            {                if (pair.Value == null)                {                    Log.Error(this.GetType().ToString(), "WxPayData内部含有值为null的字段!");                    throw new WxPayException("WxPayData内部含有值为null的字段!");                }                if (pair.Key != "sign" && pair.Value.ToString() != "")                {                    buff += pair.Key + "=" + pair.Value + "&";                }            }            buff = buff.Trim('&');            return buff;        }        /**        * @Dictionary格式化成Json         * @return json串数据        */        public string ToJson()        {            string jsonStr = JsonMapper.ToJson(m_values);            return jsonStr;        }        /**        * @values格式化成能在Web页面上显示的结果(因为web页面上不能直接输出xml格式的字符串)        */        public string ToPrintStr()        {            string str = "";            foreach (KeyValuePair pair in m_values)            {                if (pair.Value == null)                {                    Log.Error(this.GetType().ToString(), "WxPayData内部含有值为null的字段!");                    throw new WxPayException("WxPayData内部含有值为null的字段!");                }                str += string.Format("{0}={1}
", pair.Key, pair.Value.ToString()); } Log.Debug(this.GetType().ToString(), "Print in Web Page : " + str); return str; } /** * @生成签名,详见签名生成算法 * @return 签名, sign字段不参加签名 */ public string MakeSign() { //转url格式 string str = ToUrl(); //在string后加入API KEY str += "&key=" + WxPayConfig.KEY; //MD5加密 var md5 = MD5.Create(); var bs = md5.ComputeHash(Encoding.UTF8.GetBytes(str)); var sb = new StringBuilder(); foreach (byte b in bs) { sb.Append(b.ToString("x2")); } //所有字符转为大写 return sb.ToString().ToUpper(); } /** * * 检测签名是否正确 * 正确返回true,错误抛异常 */ public bool CheckSign() { //如果没有设置签名,则跳过检测 if (!IsSet("sign")) { Log.Error(this.GetType().ToString(), "WxPayData签名存在但不合法!"); throw new WxPayException("WxPayData签名存在但不合法!"); } //如果设置了签名但是签名为空,则抛异常 else if(GetValue("sign") == null || GetValue("sign").ToString() == "") { Log.Error(this.GetType().ToString(), "WxPayData签名存在但不合法!"); throw new WxPayException("WxPayData签名存在但不合法!"); } //获取接收到的签名 string return_sign = GetValue("sign").ToString(); //在本地计算新的签名 string cal_sign = MakeSign(); if (cal_sign == return_sign) { return true; } Log.Error(this.GetType().ToString(), "WxPayData签名验证错误!"); throw new WxPayException("WxPayData签名验证错误!"); } /** * @获取Dictionary */ public SortedDictionary GetValues() { return m_values; } }

WxPayData

配置文件信息

    /**    *     配置账号信息    */    public class WxPayConfig    {        //=======【基本信息设置】=====================================        /* 微信公众号信息配置        * APPID:绑定支付的APPID(必须配置)        * MCHID:商户号(必须配置)        * KEY:商户支付密钥,参考开户邮件设置(必须配置)        * APPSECRET:公众帐号secert(仅JSAPI支付的时候需要配置)        */        public const string APPID = "wx14e3e56f3";        public const string MCHID = "12352";        public const string KEY = "BB6BE71D7CED49A79409C9";        public const string APPSECRET = "76eb33f66129692da1624f1";        //=======【证书路径设置】=====================================         /* 证书路径,注意应该填写绝对路径(仅退款、撤销订单时需要)        */        public const string SSLCERT_PATH = "cert/apiclient_cert.p12";        public const string SSLCERT_PASSWORD = "123502";        //=======【支付结果通知url】=====================================         /* 支付结果通知回调url,用于商户接收支付结果        */        public const string NOTIFY_URL = "http://www.baidu.com/ResultPay.aspx";        //=======【商户系统后台机器IP】=====================================         /* 此参数可手动配置也可在程序中自动获取        */        public const string IP = "150.24.91.151";        //=======【代理服务器设置】===================================        /* 默认IP和端口号分别为0.0.0.0和0,此时不开启代理(如有需要才设置)        */        public const string PROXY_URL = "http://10.152.18.220:8080";        //=======【上报信息配置】===================================        /* 测速上报等级,0.关闭上报; 1.仅错误时上报; 2.全量上报        */        public const int REPORT_LEVENL = 1;        //=======【日志级别】===================================        /* 日志等级,0.不输出日志;1.只输出错误信息; 2.输出错误和正常信息; 3.输出错误信息、正常信息和调试信息        */        public const int LOG_LEVENL =3;    }

WxPayConfig

接着我们在看2行关键的代码:

                            WxPayData unifiedOrderResult == jsApiPay.GetJsApiParameters();

此时如果wxJsApiParam变量能够顺利拿到值,那么我们前台页面的:<%=wxJsApiParam%>z这里就可以获取到我们要传递的参数,这时候就可以调用微信支付的接口,打开我们的付款页面如图所示:

               //调用微信JS api 支付               function jsApiCall()               {                   WeixinJSBridge.invoke(                   'getBrandWCPayRequest',                   <%=wxJsApiParam%>,//josn串                    function (res)                    {                      if (res.err_msg == "get_brand_wcpay_request:ok")                       {                           var OrderId=$("#OrderId").val();                           var orderProductName=$("#orderProductName").val();                           var orderMoneySum=$("#orderMoneySum").val();                             window.location.href="http://www.baodu.com/PaySkip.aspx?OrderId="+OrderId+"&orderMoneySum="+orderMoneySum+"&orderProductName="+orderProductName;                        }else                        {                          WeixinJSBridge.call('closeWindow');                        }                                              }                    );               }

(JsApiPay.cs)得到下单结果:

        public WxPayData GetUnifiedOrderResult(string body)        {            //统一下单            WxPayData data = new WxPayData();            data.SetValue("body", body);            data.SetValue("out_trade_no", WxPayApi.GenerateOutTradeNo());            data.SetValue("total_fee", total_fee);            data.SetValue("trade_type", "JSAPI");            data.SetValue("openid", openid);            WxPayData result = WxPayApi.UnifiedOrder(data);            if (!result.IsSet("appid") || !result.IsSet("prepay_id") || result.GetValue("prepay_id").ToString() == "")            {                Log.Error(this.GetType().ToString(), "UnifiedOrder response error!");                throw new WxPayException("UnifiedOrder response error!");            }            unifiedOrderResult = result;            return result;        }

(WxPayApi.cs)统一下单接口:

        /**        *         * 统一下单        * @param WxPaydata inputObj 提交给统一下单API的参数        * @param int timeOut 超时时间        * @throws WxPayException        * @return 成功时返回,其他抛异常        */        public static WxPayData UnifiedOrder(WxPayData inputObj, int timeOut = 6)        {            string url = "https://api.mch.weixin.qq.com/pay/unifiedorder";            //检测必填参数            if (!inputObj.IsSet("out_trade_no"))            {                throw new WxPayException("缺少统一支付接口必填参数out_trade_no!");            }            else if (!inputObj.IsSet("body"))            {                throw new WxPayException("缺少统一支付接口必填参数body!");            }            else if (!inputObj.IsSet("total_fee"))            {                throw new WxPayException("缺少统一支付接口必填参数total_fee!");            }            else if (!inputObj.IsSet("trade_type"))            {                throw new WxPayException("缺少统一支付接口必填参数trade_type!");            }            //关联参数            if (inputObj.GetValue("trade_type").ToString() == "JSAPI" && !inputObj.IsSet("openid"))            {                throw new WxPayException("统一支付接口中,缺少必填参数openid!trade_type为JSAPI时,openid为必填参数!");            }            if (inputObj.GetValue("trade_type").ToString() == "NATIVE" && !inputObj.IsSet("product_id"))            {                throw new WxPayException("统一支付接口中,缺少必填参数product_id!trade_type为JSAPI时,product_id为必填参数!");            }            //异步通知url未设置,则使用配置文件中的url            if (!inputObj.IsSet("notify_url"))            {                inputObj.SetValue("notify_url", WxPayConfig.NOTIFY_URL);//异步通知url            }            inputObj.SetValue("appid", WxPayConfig.APPID);//公众账号ID            inputObj.SetValue("mch_id", WxPayConfig.MCHID);//商户号            inputObj.SetValue("spbill_create_ip", WxPayConfig.IP);//终端ip                          inputObj.SetValue("nonce_str", GenerateNonceStr());//随机字符串            //签名            inputObj.SetValue("sign", inputObj.MakeSign());            string xml = inputObj.ToXml();            var start = DateTime.Now;            string response = HttpService.Post(xml, url, false, timeOut);            var end = DateTime.Now;            int timeCost = (int)((end - start).TotalMilliseconds);            WxPayData result = new WxPayData();            result.FromXml(response);            ReportCostTime(url, timeCost, result);//测速上报            return result;        }

四.最终开发的效果

五.微信公众号开发系列导航

1.0初始微信公众号

2.0创建自定义菜单

3.0查询自定义菜单

4.0公众号消息处理

5.0微信支付

6.0模板消息

以上就是.NET微信公众号开发(5.0微信支付)实例详解的详细内容,更多请关注php中文网其它相关文章!

工具库

本文由欣才IT学院整理发布,未经许可,禁止转载。
支持37不支持0