解析微信支付的實(shí)現(xiàn)方法(.NET版)
前段時(shí)間做了網(wǎng)頁(yè)版微信支付,遇到很多問(wèn)題,不過(guò)最終還是解決了,現(xiàn)在在這里記錄下開(kāi)發(fā)流程以及說(shuō)明,給其他人一些參考。
一、準(zhǔn)備工作
首先肯定得先要開(kāi)通微信支付功能,之前開(kāi)通微信支付需要三萬(wàn)的押金的,現(xiàn)在不需要了,所以就做了這個(gè)功能。
要進(jìn)行微信支付開(kāi)發(fā),需要在公眾號(hào)后臺(tái)和微信商戶后臺(tái)進(jìn)行相關(guān)的設(shè)置。
1、開(kāi)發(fā)目錄配置
微信支付需要在公眾號(hào)后臺(tái)(微信支付=》開(kāi)發(fā)配置)進(jìn)行配置支付授權(quán)目錄。這里授權(quán)目錄需要是線上地址,也就是可以通過(guò)互聯(lián)網(wǎng)訪問(wèn)到的地址,微信支付系統(tǒng)需要能夠通過(guò)互聯(lián)網(wǎng)訪問(wèn)到你的地址。
微信授權(quán)目錄需要精確到二級(jí)或三級(jí)目錄,事例:假如發(fā)起支付的鏈接是 http://www.hxfspace.net/weixin/WeXinPay/WeXinPayChoose 那么配置的目錄應(yīng)該是http://www.hxfspace.net/weixin/WeXinPay/ 其中 http://www. hxfspace.net是域名weixin是虛擬目錄 WeXinPay也就是Controller 相關(guān)的支付請(qǐng)求都在WeXinPay中的action里面。
2、OAuth2.0網(wǎng)頁(yè)授權(quán)域名設(shè)置
微信支付的時(shí)候會(huì)對(duì)支付請(qǐng)求進(jìn)行回調(diào)來(lái)獲取授權(quán)代碼(code),所以需要在這里設(shè)置授權(quán)域名。當(dāng)然這里域名是要和支付授權(quán)目錄中的域名是同一個(gè)。這個(gè)不要忘記設(shè)置了我當(dāng)時(shí)就是忘記設(shè)置然后找半天原因,哭死。
3、相關(guān)參數(shù)準(zhǔn)備
調(diào)用微信支付需要通過(guò)腳本向微信支付系統(tǒng)發(fā)起支付請(qǐng)求,參數(shù)說(shuō)明見(jiàn)微信官網(wǎng)支付平臺(tái)https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=7_7&index=6
其中package和paySign的生成需要開(kāi)發(fā)者密鑰AppSecret(應(yīng)用密鑰)、微信商戶號(hào)、微信支付密鑰,這些參數(shù)的獲取和設(shè)置可以看這篇文章http://www.dbjr.com.cn/softjc/346871.html
二、開(kāi)發(fā)流程
廢話不多說(shuō)直接說(shuō)整理之后的流程:
1、通過(guò)微信授權(quán)回調(diào)來(lái)獲取授權(quán)code
2、通過(guò)授權(quán)code來(lái)?yè)Q取網(wǎng)頁(yè)授權(quán)access_token 和openid
3、調(diào)用統(tǒng)一下單接口獲取預(yù)支付prepayId
4、組建jsapi微信支付請(qǐng)求參數(shù),發(fā)起支付
5、接收微信支付回調(diào)進(jìn)行后續(xù)操作
三、具體開(kāi)發(fā)(上代碼)
微信支付只能在線上環(huán)境中進(jìn)行,調(diào)式很不方便,所在在剛開(kāi)始開(kāi)發(fā)的時(shí)候最好在每個(gè)關(guān)鍵位置記錄好日志。
1、通過(guò)微信授權(quán)回調(diào)來(lái)獲取授權(quán)code
首先把發(fā)起支付地址以及相關(guān)參數(shù)傳給微信支付接口,微信支付接收驗(yàn)證成功之后,會(huì)重新請(qǐng)求你的支付地址并帶上授權(quán)code。
比如我這里
//判斷是否網(wǎng)頁(yè)授權(quán),獲取授權(quán)code,沒(méi)有代表沒(méi)有授權(quán),構(gòu)造網(wǎng)頁(yè)授權(quán)獲取code,并重新請(qǐng)求
if (string.IsNullOrEmpty(Request.QueryString["code"]))
{
string redirectUrl = _weChatPaySerivce.GetAuthorizeUrl(account.AppId, account.RedquestUrl,
"STATE" + "#wechat_redirect", "snsapi_base");
return Redirect(redirectUrl);
}
拼接微信網(wǎng)頁(yè)授權(quán)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);
/* 這一步發(fā)送之后,客戶會(huì)得到授權(quán)頁(yè)面,無(wú)論同意或拒絕,都會(huì)返回redirectUrl頁(yè)面。
* 如果用戶同意授權(quán),頁(yè)面將跳轉(zhuǎn)至 redirect_uri/?code=CODE&state=STATE。這里的code用于換取access_token(和通用接口的access_token不通用)
* 若用戶禁止授權(quán),則重定向后不會(huì)帶上code參數(shù),僅會(huì)帶上state參數(shù)redirect_uri?state=STATE
*/
AppLog.Write("獲取到授權(quán)url:", AppLog.LogMessageType.Debug);
return url;
}
2、通過(guò)授權(quán)code來(lái)?yè)Q取網(wǎng)頁(yè)授權(quán)access_token 和openid
從第一步中獲取到授權(quán)code之后,組合網(wǎng)頁(yè)授權(quán)請(qǐng)求url,來(lái)獲取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 請(qǐng)求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、調(diào)用統(tǒng)一下單接口獲取預(yù)支付prepayId
這里RequestHandler是用的網(wǎng)上別人封裝好的dll,幫你封裝好了簽名的生成以及一些驗(yàn)證請(qǐng)求。dll可以在這他們官網(wǎng)下載http://weixin.senparc.com/
//創(chuàng)建支付應(yīng)答對(duì)象
RequestHandler packageReqHandler = new RequestHandler(null);
//初始化
packageReqHandler.Init();
//時(shí)間戳
string timeStamp = TenPayUtil.GetTimestamp();
//隨機(jī)字符串
string nonceStr = TenPayUtil.GetNoncestr();
//設(shè)置package訂單參數(shù) 生成prepayId預(yù)支付Id
packageReqHandler.SetParameter("appid", account.AppId); //公眾賬號(hào)ID
packageReqHandler.SetParameter("mch_id", account.PartnertId); //商戶號(hào)
packageReqHandler.SetParameter("nonce_str", nonceStr); //隨機(jī)字符串
packageReqHandler.SetParameter("body", account.Body);
packageReqHandler.SetParameter("out_trade_no", account.OrderSerialId); //商家訂單號(hào)
packageReqHandler.SetParameter("total_fee", account.TotalAmount); //商品金額,以分為單位(money * 100).ToString()
packageReqHandler.SetParameter("spbill_create_ip", account.RequestIp); //用戶的公網(wǎng)ip,不是商戶服務(wù)器IP
packageReqHandler.SetParameter("notify_url", account.NotifyUrl); //接收財(cái)付通通知的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("調(diào)用統(tǒng)一下單接口,下單結(jié)果:--"+result+"請(qǐng)求參數(shù):"+data);
var res = XDocument.Parse(result);
prepayId = res.Element("xml").Element("prepay_id").Value;
AppLog.Write("調(diào)用統(tǒng)一下單接口獲取預(yù)支付prepayId成功", AppLog.LogMessageType.Debug);
}
catch (Exception ex)
{
AppLog.Write("獲取到openid和access_tokenu異常", AppLog.LogMessageType.Debug, ex);
MailHelp.SendMail("調(diào)用統(tǒng)一下單接口獲取預(yù)支付prepayid異常:", ex);
return null;
}
4、組建jsapi微信支付請(qǐng)求參數(shù),發(fā)起支付
我這里是首先組裝好微信支付所需要的參數(shù),然后再創(chuàng)建調(diào)用js腳本
//生成JsAPI支付參數(shù)
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"
};
創(chuàng)建調(diào)用腳本
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("生成可執(zhí)行腳本成功", AppLog.LogMessageType.Debug);
return js;
}
5、接收微信支付回調(diào)進(jìn)行后續(xù)操作
回調(diào)的時(shí)候首先需要驗(yàn)證簽名是否正確,保證安全性,簽名驗(yàn)證通過(guò)之后再進(jìn)行后續(xù)的操作,訂單狀態(tài)、通知啥的。
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();
這里有一點(diǎn)需要注意,就是微信支付回調(diào)的時(shí)候微信會(huì)通知八次,好像是這個(gè)數(shù)吧,所以你需要在第一次收到通知之后,把收到請(qǐng)求這個(gè)狀態(tài)以xml的格式響應(yīng)給微信支付接口。當(dāng)然你不進(jìn)行這個(gè)操作也是可以的,再回調(diào)的時(shí)候 每次去判斷該訂單是否已經(jīng)回調(diào)成功,回調(diào)成功則不進(jìn)行處理就可以了。
原文鏈接:http://www.cnblogs.com/minesnil-forfaith/p/4976006.html
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
asp.net LINQ中數(shù)據(jù)庫(kù)連接字符串的問(wèn)題
這兩天一直在用LINQ做開(kāi)發(fā),也是第一次嘗試用LINQ做開(kāi)發(fā),效率沒(méi)的說(shuō),開(kāi)發(fā)過(guò)程中遇到一個(gè)問(wèn)題困擾了我好久,今天問(wèn)題終于解決了,發(fā)上來(lái)和大家分享一下,也給自己做個(gè)備忘。2010-03-03
高仿Windows Phone QQ登錄界面實(shí)例代碼
這篇文章主要介紹了高仿Windows Phone QQ登錄界面實(shí)例代碼,有需要的朋友可以參考一下2013-12-12
詳解如何在ASP.NET Core Web API中以三種方式返回?cái)?shù)據(jù)
這篇文章主要介紹了詳解如何在ASP.NET Core Web API中以三種方式返回?cái)?shù)據(jù),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01
.NET Core開(kāi)發(fā)日志之OData(Open Data Protocol)
這篇文章主要給大家介紹了關(guān)于.NET Core開(kāi)發(fā)日志之OData(Open Data Protocol)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02
Visual Studio Debugger七個(gè)鮮為人知的小功能
這篇文章主要為大家詳細(xì)介紹了Visual Studio Debugger七個(gè)鮮為人知的小功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06
.Net?Core3.0?WebApi?項(xiàng)目框架搭建之使用Serilog替換掉Log4j
Serilog 是一個(gè)用于.NET應(yīng)用程序的日志記錄開(kāi)源庫(kù),配置簡(jiǎn)單,接口干凈,并可運(yùn)行在最新的.NET平臺(tái)上,這篇文章主要介紹了.Net?Core3.0?WebApi?項(xiàng)目框架搭建之使用Serilog替換掉Log4j,需要的朋友可以參考下2022-02-02
asp.net下xml當(dāng)作導(dǎo)航數(shù)據(jù)源實(shí)現(xiàn)動(dòng)態(tài)權(quán)限
如果有權(quán)限的話 可以通過(guò)節(jié)點(diǎn)的Roles屬性判斷當(dāng)前登陸的賬號(hào)角色名是否符合然后判斷輸出這樣的話您就可以直接操作XML數(shù)據(jù) 而不用考慮別的。2009-12-12
ASP.NET Core Controller與IOC結(jié)合問(wèn)題整理
在本篇文章里小編給大家整理了一篇關(guān)于ASP.NET Core Controller與IOC結(jié)合問(wèn)題整理內(nèi)容,有需要的朋友們可以學(xué)習(xí)下。2021-01-01

