亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

.NET微信公眾號開發-5.0微信支付

2019-11-14 15:55:45
字體:
來源:轉載
供稿:網友

一.前言

在開始做這個功能之前,我們要做的第一件事情就是思考,如何做這個微信支付,從哪里開始,從哪里入手,官方的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><body>    <div>              <br />       <br />       <br />       <input  type="hidden" id="OrderId" name="OrderId" value="<%=OrderId %>"/>       <input  type="hidden" id="orderMoneySum" name="orderMoneySum" value="<%=orderMoneySum %>"/>       <input  type="hidden" id="orderProductName" name="orderProductName" value="<%=orderProductName %>"/>       <span class="fLeft" style="font-size:20px;color:Purple">&nbsp;&nbsp;您確認付款<label style="font-size:25px;color:Red"><%=Money%></label>元...</span>      <div><button type="button" class="btn-pay" title="確認支付" onclick="callpay()">立即支付</button></div>    </div></body></html>

需要注意的是:

微信JS API只能在微信內置瀏覽器中使用,其他瀏覽器調用無效。微信提供getBrandWCPayRequest接口供商戶前端網頁調用,調用之前微信會鑒定商戶支付權限,若商戶具有調起支付的權限,則將開始支付流程。 這里主要介紹支付前的接口調用規則,支付狀態消息通知機制請參加下文。接口需要注意:所有傳入參數都是字符串類型!

 

getBrandWCPayRequest參數如表6-5所示。

參數

名稱

必填

格式

說明

appId

公眾號id

字符串類型

商戶注冊具有支付權限的公眾號成功后即可獲得;

timeStamp

時間戳

字符串類型,32個字節以下

商戶生成,從1970年1月1日00:00:00至今的秒數,即當前的時間,且最終需要轉換為字符串形式;

nonceStr

隨機字符串

字符串類型,32個字節以下

商戶生成的隨機字符串;

package

訂單詳情擴展字符串

字符串類型,4096個字節以下

商戶將訂單信息組成該字符串,具體組成方案參見接口使用說明中package組包幫劣;由商戶按照規范拼接后傳入;

signType

簽名方式

字符串類型,參數取值"SHA1"

按照文檔中所示填入,目前僅支持SHA1;

paySign

簽名

字符串類型

商戶將接口列表中的參數按照指定方式迚行簽名,簽名方式使用signType中標示的簽名方式,具體簽名方案參見接口使用說明中簽名幫劣;由商戶按照規范簽名后傳入;

表6-5 getBrandWCPayRequest參數

    getBrandWCPayRequest返回值如表6-6所示。

返回值

說明

err_msg

get_brand_wcpay_request:ok  支付成功
get_brand_wcpay_request:cancel 支付過程中用戶取消
get_brand_wcpay_request:fail 支付失敗

表6-6 getBrandWCPayRequest返回值

JS API的返回結果 get_brand_wcpay_request:ok 僅在用戶成功完成支付時返回。由于前端交互復雜,get_brand_wcpay_request:cancel 或者 get_brand_wcpay_request:fail 可以統一處理為用戶遇到錯誤或者主動放棄,不必細化區分。

 pay.aspx后臺頁面代碼:

    /// <summary>    /// 微信支付核心頁面    /// </summary>    public partial class Pay : System.Web.UI.Page    {        //post提交中Request.Form取不到值        public string wxJsApiParam { get; set; } //H5調起JS API參數        public string Money { get; set; }        public string OrderId { get; set; }        public string orderMoneySum { get; set; }//商品金額        public string orderProductName { get; set; }//商品名稱        protected void Page_Load(object sender, EventArgs e)        {            if (!IsPostBack)            {                    JsApiPay jsApiPay = new JsApiPay(this);                    try                    {                        string total_fee = Request["total_fee"];                        orderMoneySum = total_fee;                        string ParkName = Request["ParkName"];                        orderProductName = ParkName+"停車費用";                        OrderId = Request["OrderId"];                        if (string.IsNullOrWhiteSpace(total_fee)||total_fee=="0")                        {                            throw new WxPayException("<span style='color:#FF0000;font-size:20px'>" + "費用為零,請求參數錯誤" + "</span>");                        }                        jsApiPay.total_fee =int.Parse((Convert.ToDouble(total_fee)*100).ToString());                        Money = (Convert.ToDouble(jsApiPay.total_fee)/100).ToString();                        jsApiPay.orderid = OrderId;                        //JSAPI支付預處理                        try                        {
Common common = new Common(Context); jsApiPay.openid
= common.GetOpenId(); if (Common.OpenId == "Openid") { throw new WxPayException("OpenId為空無法下單!"); } jsApiPay.access_token = Common.access_token; WxPayData unifiedOrderResult = jsApiPay.GetUnifiedOrderResult(ParkName); wxJsApiParam = jsApiPay.GetJsApiParameters();//獲取H5調起JS API參數 } catch (Exception ex) { Response.Write("<span style='color:#FF0000;font-size:20px'>" + "下單失敗,請返回重試:" + ex.InnerException.Message + "</span>"); } } catch (Exception ex) { Response.Write("<span style='color:#FF0000;font-size:20px'>" + "頁面加載出錯,請重試:" + ex.Message + "</span>"); } } } }

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

在支付的時候,我們需要首先獲取用戶的openId,然而獲取用戶openId的這個過程我們首先要進行Oauth2認證,在官方的demo中提供了JsApiPay.cs這個核心類庫,里面已經有這個GetOpenidAndAccessToken()方法 。我這里是拿過來寫成了自己的一個公共幫助類。Common.cs

    /// <summary>    /// 公共幫助類    /// </summary>    public class Common    {        private  HttpContext Context { get; set; }        public static string OpenId = "Openid";        public static string access_token = "access_token";        #region 構造函數        /// <summary>        /// 構造函數        /// </summary>        /// <param name="Context"></param>        public Common(HttpContext context)        {            this.Context = context;        }         #endregion        #region 通過code換取AccessToken        /// <summary>        /// 通過code換取AccessToken        /// </summary>        public  void GetOpenidAndAccessToken()        {            if (!string.IsNullOrEmpty(Context.Request.QueryString["code"]))            {                //獲取code碼,以獲取openid和access_token                string code = Context.Request.QueryString["code"];                GetOpenidAndAccessTokenFromCode(code);            }            else            {                //構造網頁授權獲取code的URL                string host = Context.Request.Url.Host;                string path = Context.Request.Path;                string redirect_uri = HttpUtility.UrlEncode("http://" + host + path);                WxPayData data = new WxPayData();                data.SetValue("appid", WxPayConfig.APPID);                data.SetValue("redirect_uri", redirect_uri);                data.SetValue("response_type", "code");                data.SetValue("scope", "snsapi_base");                data.SetValue("state", "STATE" + "#wechat_redirect");                string url = "https://open.weixin.QQ.com/connect/oauth2/authorize?" + data.ToUrl();                try                {                    //觸發微信返回code碼                             Context.Response.Redirect(url);//Redirect函數會拋出ThreadAbortException異常,不用處理這個異常                }                catch (System.Threading.ThreadAbortException ex)                {                }            }        }                #endregion        #region 通過用戶授權獲取AccessToken和OpenId        /// <summary>        /// 通過用戶授權獲取AccessToken和OpenId        /// </summary>        /// <param name="code"></param>        public  void GetOpenidAndAccessTokenFromCode(string code)        {            try            {                //構造獲取openid及access_token的url                WxPayData data = new WxPayData();                data.SetValue("appid", WxPayConfig.APPID);                data.SetValue("secret", WxPayConfig.APPSECRET);                data.SetValue("code", code);                data.SetValue("grant_type", "authorization_code");                string url = "https://api.weixin.qq.com/sns/oauth2/access_token?" + data.ToUrl();                //請求url以獲取數據                string result = HttpService.Get(url);                //保存access_token,用于收貨地址獲取                JsonData jd = JsonMapper.ToObject(result);                access_token = (string)jd["access_token"];                //獲取用戶openid                OpenId = (string)jd["openid"];            }            catch (Exception ex)            {                throw new WxPayException(ex.ToString());            }        }         #endregion        #region 獲取OpenId        /// <summary>        /// 獲取OpenId        /// </summary>        /// <param name="postStr"></param>        /// <returns></returns>        public string GetOpenId()        {                Common common = new Common(Context);                common.GetOpenidAndAccessToken();                return OpenId;        }         #endregion    }

 

 

 public class JsApiPay    {        /// <summary>        /// 保存頁面對象,因為要在類的方法中使用Page的Request對象        /// </summary>        private Page page {get;set;}        /// <summary>        /// openid用于調用統一下單接口        /// </summary>        public string openid { get; set; }        /// <summary>        /// access_token用于獲取收貨地址js函數入口參數        /// </summary>        public string access_token { get; set; }        /// <summary>        /// 商品金額,用于統一下單        /// </summary>        public int total_fee { get; set; }        /// <summary>        /// 訂單Id        /// </summary>        public string orderid { get; set; }        /// <summary>        /// 統一下單接口返回結果        /// </summary>        public WxPayData unifiedOrderResult { get; set; }         public JsApiPay(Page page)        {            this.page = page;        }        /**        *         * 網頁授權獲取用戶基本信息的全部過程        * 詳情請參看網頁授權獲取用戶基本信息:http://mp.weixin.qq.com/wiki/17/c0f37d5704f0b64713d5d2c37b468d75.html        * 第一步:利用url跳轉獲取code        * 第二步:利用code去獲取openid和access_token        *         */        public void GetOpenidAndAccessToken()        {            if (!string.IsNullOrEmpty(page.Request.QueryString["code"]))            {                //獲取code碼,以獲取openid和access_token                string code = page.Request.QueryString["code"];                //Log.Debug(this.GetType().ToString(), "Get code : " + code);                GetOpenidAndAccessTokenFromCode(code);            }            else            {                //構造網頁授權獲取code的URL                string host = page.Request.Url.Host;                //Log.Debug(this.GetType().ToString(), "host" + host);                string path = page.Request.Path;                string redirect_uri = HttpUtility.UrlEncode("http://" + host + path);                                WxPayData data = new WxPayData();                data.SetValue("appid", WxPayConfig.APPID);                data.SetValue("redirect_uri", redirect_uri);                data.SetValue("response_type", "code");                data.SetValue("scope", "snsapi_base");                data.SetValue("state", "STATE" + "#wechat_redirect");                string url = "https://open.weixin.qq.com/connect/oauth2/authorize?" + data.ToUrl();                Log.Debug(this.GetType().ToString(), "Will Redirect to URL : " + url);                try                {                    //觸發微信返回code碼                             page.Response.Redirect(url);//Redirect函數會拋出ThreadAbortException異常,不用處理這個異常                }                catch(System.Threading.ThreadAbortException ex)                {                }            }        }        /**        *         * 通過code換取網頁授權access_token和openid的返回數據,正確時返回的JSON數據包如下:        * {        *  "access_token":"ACCESS_TOKEN",        *  "expires_in":7200,        *  "refresh_token":"REFRESH_TOKEN",        *  "openid":"OPENID",        *  "scope":"SCOPE",        *  "unionid": "o6_bmasdasdsad6_2sgVt7hMZOPfL"        * }        * 其中access_token可用于獲取共享收貨地址        * openid是微信支付jsapi支付接口統一下單時必須的參數        * 更詳細的說明請參考網頁授權獲取用戶基本信息:http://mp.weixin.qq.com/wiki/17/c0f37d5704f0b64713d5d2c37b468d75.html        * @失敗時拋異常WxPayException        */        public void GetOpenidAndAccessTokenFromCode(string code)        {            try            {                //構造獲取openid及access_token的url                WxPayData data = new WxPayData();                data.SetValue("appid", WxPayConfig.APPID);                data.SetValue("secret", WxPayConfig.APPSECRET);                data.SetValue("code", code);                data.SetValue("grant_type", "authorization_code");                string url = "https://api.weixin.qq.com/sns/oauth2/access_token?" + data.ToUrl();                //請求url以獲取數據                string result = HttpService.Get(url);                //Log.Debug(this.GetType().ToString(), "GetOpenidAndAccessTokenFromCode response : " + result);                //保存access_token,用于收貨地址獲取                JsonData jd = JsonMapper.ToObject(result);                access_token = (string)jd["access_token"];                //獲取用戶openid                openid = (string)jd["openid"];                //Log.Debug(this.GetType().ToString(), "Get openid : " + openid);                //Log.Debug(this.GetType().ToString(), "Get access_token : " + access_token);            }            catch (Exception ex)            {                Log.Error(this.GetType().ToString(), ex.ToString());                throw new WxPayException(ex.ToString());            }        }        /**         * 調用統一下單,獲得下單結果         * @return 統一下單結果         * @失敗時拋異常WxPayException         */        public WxPayData GetUnifiedOrderResult()        {            //統一下單            WxPayData data = new WxPayData();            data.SetValue("body", "test");            //data.SetValue("attach", "test");            data.SetValue("out_trade_no", WxPayApi.GenerateOutTradeNo());            data.SetValue("total_fee", total_fee);            data.SetValue("time_start", DateTime.Now.ToString("yyyyMMddHHmmss"));            data.SetValue("time_expire", DateTime.Now.AddMinutes(10).ToString("yyyyMMddHHmmss"));            //data.SetValue("goods_tag", "test");            data.SetValue("trade_type", "JSAPI");            data.SetValue("openid", openid);            data.SetValue("orderid", orderid);            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;        }        public WxPayData GetUnifiedOrderResult(string body)        {            //統一下單            WxPayData data = new WxPayData();            data.SetValue("body", body);            //data.SetValue("attach", "test");            data.SetValue("out_trade_no", WxPayApi.GenerateOutTradeNo());            data.SetValue("total_fee", total_fee);            //data.SetValue("time_start", DateTime.Now.ToString("yyyyMMddHHmmss"));            //data.SetValue("time_expire", DateTime.Now.AddMinutes(10).ToString("yyyyMMddHHmmss"));            //data.SetValue("goods_tag", "test");            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;        }        /**        *          * 從統一下單成功返回的數據中獲取微信瀏覽器調起jsapi支付所需的參數,        * 微信瀏覽器調起JSAPI時的輸入參數格式如下:        * {        *   "appId" : "wx2421b1c4370ec43b",     //公眾號名稱,由商戶傳入             *   "timeStamp":" 1395712654",         //時間戳,自1970年以來的秒數             *   "nonceStr" : "e61463f8efa94090b1f366cccfbbb444", //隨機串             *   "package" : "prepay_id=u802345jgfjsdfgsdg888",             *   "signType" : "md5",         //微信簽名方式:            *   "paySign" : "70EA570631E4BB79628FBCA90534C63FF7FADD89" //微信簽名         * }        * @return string 微信瀏覽器調起JSAPI時的輸入參數,json格式可以直接做參數用        * 更詳細的說明請參考網頁端調起支付API:http://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_7        *         */        public string GetJsApiParameters()        {            WxPayData jsApiParam = new WxPayData();            jsApiParam.SetValue("appId", unifiedOrderResult.GetValue("appid"));            jsApiParam.SetValue("timeStamp", WxPayApi.GenerateTimeStamp());            jsApiParam.SetValue("nonceStr", WxPayApi.GenerateNonceStr());            jsApiParam.SetValue("package", "prepay_id=" + unifiedOrderResult.GetValue("prepay_id"));            jsApiParam.SetValue("signType", "MD5");            jsApiParam.SetValue("paySign", jsApiParam.MakeSign());            string parameters = jsApiParam.ToJson();            return parameters;        }        /**        *         * 獲取收貨地址js函數入口參數,詳情請參考收貨地址共享接口:http://pay.weixin.qq.com/wiki/doc/api/jsapi.php?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官方都有相應的代碼.

 /// <summary>    /// 微信支付協議接口數據類,所有的API接口通信都依賴這個數據結構,    /// 在調用接口之前先填充各個字段的值,然后進行接口通信,    /// 這樣設計的好處是可擴展性強,用戶可隨意對協議進行更改而不用重新設計數據結構,    /// 還可以隨意組合出不同的協議數據包,不用為每個協議設計一個數據包結構    /// </summary>    public class WxPayData    {        public WxPayData()        {        }        //采用排序的Dictionary的好處是方便對數據包進行簽名,不用再簽名之前再做一次排序        private SortedDictionary<string, object> m_values = new SortedDictionary<string, object>();        /**        * 設置某個字段的值        * @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 = "<xml>";            foreach (KeyValuePair<string, object> 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 + "</" + pair.Key + ">";                }                else if (pair.Value.GetType() == typeof(string))                {                    xml += "<" + pair.Key + ">" + "<![CDATA[" + pair.Value + "]]></" + pair.Key + ">";                }                else//除了string和int類型不能含有其他數據類型                {                    Log.Error(this.GetType().ToString(), "WxPayData字段數據類型錯誤!");                    throw new WxPayException("WxPayData字段數據類型錯誤!");                }            }            xml += "</xml>";            return xml;        }        /**        * @將xml轉為WxPayData對象并返回對象內部的數據        * @param string 待轉換的xml串        * @return 經轉換得到的Dictionary        * @throws WxPayException        */        public SortedDictionary<string, object> 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;//獲取到根節點<xml>            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<string, object> 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<string, object> 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}<br>", 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<string, object> 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.GetUnifiedOrderResult(ParkName);//調用jsApiPay的下單接口并且得到返回的結果。                            wxJsApiParam = jsApiPay.GetJsApiParameters();//獲取H5調起JS API參數 

此時如果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模板消息


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩美女主播视频| 国产精品电影在线观看| 久久综合免费视频影院| 亚洲人午夜精品| 成人国产精品免费视频| 日韩欧美国产黄色| 狠狠躁天天躁日日躁欧美| 88xx成人精品| 日韩久久免费电影| 亚洲免费高清视频| 国产一区二区香蕉| 成人免费xxxxx在线观看| 国产在线观看一区二区三区| 日韩大片在线观看视频| 日韩av在线一区二区| 国产精品亚洲片夜色在线| 日韩精品极品毛片系列视频| 日韩av电影免费观看高清| 亚洲综合精品一区二区| 久久久久久久久久久国产| 伊人一区二区三区久久精品| 亚洲人a成www在线影院| 国产区精品在线观看| 欧美日本在线视频中文字字幕| 欧美疯狂xxxx大交乱88av| 国产视频亚洲精品| 精品在线小视频| 伊人激情综合网| 日韩欧美一区视频| 国产精品日韩av| 亚洲三级 欧美三级| 日韩中文字幕在线视频播放| 久久久伊人日本| 日韩欧美精品网址| 亚洲精品美女免费| 伊人伊成久久人综合网小说| 欧美高清无遮挡| 欧美精品亚州精品| 亚洲精品www久久久| 国产精品久久久久久久电影| 26uuu日韩精品一区二区| 日本欧美国产在线| 精品人伦一区二区三区蜜桃网站| 日韩免费观看高清| 91精品国产综合久久久久久久久| 伊人久久五月天| 久久精品国产久精国产思思| 国产精品久在线观看| 久久久久www| 亚洲国产精品久久91精品| 久久影视电视剧免费网站| 久久天天躁狠狠躁夜夜av| 久久久久久久色| 青青久久av北条麻妃海外网| 中文字幕不卡在线视频极品| 日本不卡高字幕在线2019| 成人网在线免费看| 成人久久一区二区| 久久99久国产精品黄毛片入口| 国产精品成人免费电影| 久久伊人色综合| 中文字幕亚洲欧美在线| 在线观看久久久久久| 中文字幕久久久av一区| 久久久久久久久久久久av| 欧美性xxxxhd| 欧美日韩精品中文字幕| 国产精品av在线播放| 中文字幕av一区二区| 亚洲国产91色在线| 亚洲a在线观看| 456亚洲影院| 亚洲欧美综合精品久久成人| 欧美在线免费看| 欧美日韩激情小视频| 日韩精品视频三区| 国产欧美日韩中文| 亚洲精品国产综合区久久久久久久| 国产视频观看一区| 亚洲影院高清在线| 欧美成人剧情片在线观看| 欧美另类老女人| 欧美性猛交xxxx乱大交3| 亚洲一区二区免费| 欧美午夜性色大片在线观看| 中文字幕亚洲一区| 亚洲在线观看视频| 热久久这里只有| 欧美极品少妇与黑人| 国产日韩精品在线观看| 欧美多人乱p欧美4p久久| 欧美日韩福利在线观看| 精品福利在线看| 51精品国产黑色丝袜高跟鞋| 日本欧美中文字幕| 55夜色66夜色国产精品视频| 久久精品国产欧美激情| 日韩欧美在线观看| 国产精品视频一区二区高潮| 夜夜嗨av一区二区三区四区| 亚洲一区二区三区在线视频| 欧美亚洲激情视频| 夜夜嗨av一区二区三区四区| 成人免费淫片aa视频免费| 久久久久久久久国产| 亚洲嫩模很污视频| 日韩欧美有码在线| 亚洲欧洲一区二区三区在线观看| 亚洲人成电影网站色xx| 69久久夜色精品国产69乱青草| 国产精品日韩欧美综合| 亚洲欧美日韩国产中文专区| 中文字幕亚洲欧美日韩高清| 一区二区三区四区在线观看视频| 中文字幕一区二区三区电影| 成人激情视频在线播放| 欧美超级乱淫片喷水| 中文字幕亚洲综合久久筱田步美| 亚洲一区av在线播放| 久久久久久久久久亚洲| 亚洲精品免费av| 亚洲欧美在线一区二区| 亚洲精品久久久久久久久久久久久| 久久69精品久久久久久久电影好| 成人午夜一级二级三级| 亚洲一区二区三区在线视频| 日韩有码在线视频| 在线观看欧美视频| 精品国产老师黑色丝袜高跟鞋| 欧美日韩国产页| 精品久久久久国产| 亚洲精品乱码久久久久久金桔影视| 91免费版网站入口| 日韩亚洲成人av在线| 欧美视频第一页| 日本成人免费在线| 毛片精品免费在线观看| 国产亚洲精品美女| 91麻豆国产精品| 精品无码久久久久久国产| 91久久国产婷婷一区二区| 日韩高清有码在线| 欧美孕妇性xx| xvideos国产精品| 一本色道久久88综合亚洲精品ⅰ| 国产日韩精品在线观看| 国产精品久久久久久久天堂| 日韩中文字幕免费| 一道本无吗dⅴd在线播放一区| 美女av一区二区三区| 国产区精品在线观看| 一本一本久久a久久精品综合小说| 正在播放国产一区| 国产激情久久久| 欧洲中文字幕国产精品| 91精品久久久久久| 91精品国产综合久久香蕉的用户体验| 国产精品美女久久久久av超清| 亚洲欧洲一区二区三区在线观看| 亚洲精品资源美女情侣酒店| 69久久夜色精品国产69| 久久露脸国产精品| 亚洲人成啪啪网站| 欧美精品少妇videofree|