C#實現(xiàn)微信退款及對賬功能的示例詳解
需求
在招聘報名系統(tǒng)里,考務(wù)費支付是其中一個環(huán)節(jié),支付方式很多種,比如銀聯(lián)、微信、支付寶等等。本次我們以微信支付進行舉例,在考生注冊賬號、編寫簡歷、報名職位、被初審核通過等一系列基礎(chǔ)的條件的具備下,可以進入支付考務(wù)費的環(huán)節(jié)(筆試費用),我們會為其生成一個支付二維碼,考生支付后(無論成功與否),都會記錄其支付結(jié)果狀態(tài)。
在實際的應(yīng)用中,對于支付成功的考生,我們會遇到實現(xiàn)退款的需求,只要包括如下場景:
1、根據(jù)政策規(guī)定,某些符合全部或部分退款條件的考生。
2、其它未知原因,重復(fù)支付訂單的考生。
3、其它不可抗力,需求進行退款的考生。
基礎(chǔ)準(zhǔn)備
在實現(xiàn)功能前,做為企業(yè),我們需要申請一個微信服務(wù)號,并成為微信支付商家。
1、申請服務(wù)號
申請成功后會獲得到 AppId 和 AppSecret 用于后續(xù)開發(fā),如關(guān)聯(lián)支付商戶、網(wǎng)頁授權(quán)登錄等。
具體指引請參照微信公眾平臺首頁:https://mp.weixin.qq.com/cgi-bin/loginpage
2、成為微信支付商家
申請成功后會獲得 Mchid 和 paySignKey 用于微信支付、退款等,請在商家后臺務(wù)必關(guān)聯(lián)申請的公眾號。
具體指引請參照微信支付平臺首頁:https://pay.weixin.qq.com/index.php/core/home/login
上述兩個平臺申請成功后,請登錄微信支付商家平臺,進行如下圖操作:

在產(chǎn)品中心、AppID帳號管理、關(guān)聯(lián) AppID(即申請的服務(wù)號)
另外一個重要配置是支付目錄,我們寫的支付程序需要在這里設(shè)置,如下圖:

關(guān)鍵代碼
操作界面
界面上會顯示最近一筆的微信訂單支付情況,包括訂單號、交費時間、交費金額、退款金額。其中退款金額不能大于成功交費金額,否則會返回失敗。另外,還可以顯示微信交易跟蹤日志列表信息,如果訂單號、交易價格、openid、返回信息、交易狀態(tài)等。
示例界面如下:

退款訂單類及方法
實現(xiàn)微信退款,需要在支付商家平臺申請退款證書,證書文件保存到自定義的目錄中,在退款時指定路徑。
退款示例代碼如下:
const string RefundOrderUrl = "https://api.mch.weixin.qq.com/secapi/pay/refund"; //退款申請API地址
const string RefundQueryUrl = "https://api.mch.weixin.qq.com/pay/refundquery"; //退款查詢API地址
//退款訂單明細(xì)類
public class RefundOrderDetail
{
/// <summary>
/// 返回狀態(tài)碼,SUCCESS/FAIL 此字段是通信標(biāo)識,非交易標(biāo)識,交易是否成功需要查看trade_state來判斷
/// </summary>
public string return_code = "";
/// <summary>
/// 返回信息返回信息,如非空,為錯誤原因 簽名失敗 參數(shù)格式校驗錯誤
/// </summary>
public string return_msg = "";
/// <summary>
/// 業(yè)務(wù)結(jié)果,SUCCESS/FAIL
/// </summary>
public string result_code = "";
/// <summary>
/// 錯誤代碼
/// </summary>
public string err_code = "";
/// <summary>
/// 錯誤代碼描述
/// </summary>
public string err_code_des = "";
/// <summary>
/// 公眾號ID(微信分配的公眾賬號 ID)
/// </summary>
public string appid = "";
/// <summary>
/// 商戶號(微信支付分配的商戶號)
/// </summary>
public string mch_id = "";
/// <summary>
/// 微信支付分配的終端設(shè)備號
/// </summary>
public string device_info = "";
/// <summary>
/// 隨機字符串,不長于32位
/// </summary>
public string nonce_str = "";
/// <summary>
/// 簽名
/// </summary>
public string sign = "";
/// <summary>
/// 微信支付訂單號
/// </summary>
public string transaction_id = "";
/// <summary>
/// 商戶系統(tǒng)的訂單號,與請求一致。
/// </summary>
public string out_trade_no = "";
public string out_refund_no = "";
public string refund_id = "";
public string refund_fee = "";
public string settlement_refund_fee = "";
/// <summary>
/// 訂單總金額,單位為分
/// </summary>
public string total_fee = "";
/// </summary>
public string settlement_total_fee = "";
public string fee_type = "";
public string cash_fee = "";
public string cash_fee_type = "";
public string cash_refund_fee = "";
public string coupon_type_0 = "";
public string coupon_refund_fee = "";
public string coupon_refund_fee_0 = "";
public string coupon_refund_count = "";
public string coupon_refund_id_0 = "";
}
//退款訂單類
public class RefundOrder
{
/// <summary>
/// 公眾號ID(微信分配的公眾賬號 ID)
/// </summary>
public string appid = "";
/// <summary>
/// 商戶號(微信支付分配的商戶號)
/// </summary>
public string mch_id = "";
/// <summary>
/// 微信支付分配的終端設(shè)備號
/// </summary>
public string device_info = "";
/// <summary>
/// 隨機字符串,不長于 32 位
/// </summary>
public string nonce_str = "";
/// <summary>
/// 簽名
public string sign = "";
public string sign_type = "";
/// <summary>
/// 商戶系統(tǒng)內(nèi)部的訂單號,32個字符內(nèi)、可包含字母,確保在商戶系統(tǒng)唯一,詳細(xì)說明
/// </summary>
public string transaction_id = "";
public string out_trade_no = "";
public string out_refund_no = "";
/// <summary>
/// 訂單總金額,單位為分,不能帶小數(shù)點
/// </summary>
public int total_fee = 0;
public int refund_fee = 0;
public string refund_fee_type = "";
public string op_user_id = "";
/// <summary>
public string refund_account = "";
/// <summary>
}
//查詢對帳訂單類
public class QueryOrder
{
/// <summary>
/// 公共號ID(微信分配的公眾賬號 ID)
/// </summary>
public string appid = "";
/// <summary>
/// 商戶號(微信支付分配的商戶號)
/// </summary>
public string mch_id = "";
/// <summary>
/// 微信訂單號,優(yōu)先使用
/// </summary>
public string transaction_id = "";
/// <summary>
/// 商戶系統(tǒng)內(nèi)部訂單號
/// </summary>
public string out_trade_no = "";
/// <summary>
/// 隨機字符串,不長于 32 位
/// </summary>
public string nonce_str = "";
/// <summary>
/// 簽名,參與簽名參數(shù):appid,mch_id,transaction_id,out_trade_no,nonce_str,key
/// </summary>
public string sign = "";
}
//申請退款方法,返回退款訂單明細(xì)類
//參數(shù)refundorder為退款訂單類, key 為支付簽名KEY,cert為證書地址,password 為證書密碼
public RefundOrderDetail getRefundOrderDetail(RefundOrder refundorder, string key,string cert,string password)
{
string post_data = getRefundOrderXml(refundorder, key);
string request_data = PostXmlAndCertToUrl(RefundOrderUrl, post_data,cert,password);
RefundOrderDetail orderdetail = new RefundOrderDetail();
SortedDictionary<string, string> requestXML = GetInfoFromXml(request_data);
foreach (KeyValuePair<string, string> k in requestXML)
{
switch (k.Key)
{
case "retuen_code":
orderdetail.result_code = k.Value;
break;
case "return_msg":
orderdetail.return_msg = k.Value;
break;
case "result_code":
orderdetail.result_code = k.Value;
break;
case "err_code":
orderdetail.err_code = k.Value;
break;
case "err_code_des":
orderdetail.err_code_des = k.Value;
break;
case "appid":
orderdetail.appid = k.Value;
break;
case "mch_id":
orderdetail.mch_id = k.Value;
break;
case "device_info":
orderdetail.device_info = k.Value;
break;
case "nonce_str":
orderdetail.nonce_str = k.Value;
break;
case "sign":
orderdetail.sign = k.Value;
break;
case "transaction_id":
orderdetail.transaction_id = k.Value;
break;
case "out_trade_no":
orderdetail.out_trade_no = k.Value;
break;
case "out_refund_no":
orderdetail.out_refund_no = k.Value;
break;
case "refund_id":
orderdetail.refund_id = k.Value;
break;
case "refund_fee":
orderdetail.refund_fee = k.Value;
break;
case "total_fee":
orderdetail.total_fee = k.Value;
break;
case "settlement_refund_fee":
orderdetail.settlement_refund_fee = k.Value;
break;
case "settlement_total_fee":
orderdetail.settlement_total_fee = k.Value;
break;
case "fee_type":
orderdetail.fee_type = k.Value;
break;
case "cash_fee":
orderdetail.cash_fee = k.Value;
break;
case "cash_fee_type ":
orderdetail.cash_fee_type = k.Value;
break;
case "cash_refund_fee":
orderdetail.cash_refund_fee = k.Value;
break;
case "coupon_type_0":
orderdetail.coupon_type_0 = k.Value;
break;
case "coupon_refund_fee":
orderdetail.coupon_refund_fee = k.Value;
break;
case "coupon_refund_fee_0":
orderdetail.coupon_refund_fee_0 = k.Value;
break;
case "coupon_refund_count":
orderdetail.coupon_refund_count = k.Value;
break;
case "coupon_refund_id_0":
orderdetail.coupon_refund_id_0 = k.Value;
break;
default:
break;
}
}
return orderdetail;
}
protected string getRefundOrderXml(RefundOrder refundorder, string key)
{
string return_string = string.Empty;
SortedDictionary<string, string> sParams = new SortedDictionary<string, string>();
sParams.Add("appid", refundorder.appid);
sParams.Add("mch_id", refundorder.mch_id);
// sParams.Add("transaction_id", refundorder.transaction_id);
sParams.Add("out_trade_no", refundorder.out_trade_no);
sParams.Add("nonce_str", refundorder.nonce_str);
sParams.Add("out_refund_no", refundorder.out_refund_no);
sParams.Add("total_fee", refundorder.total_fee.ToString());
sParams.Add("refund_fee", refundorder.refund_fee.ToString());
sParams.Add("op_user_id", refundorder.op_user_id);
refundorder.sign = getsign(sParams, key);
sParams.Add("sign", refundorder.sign);
//拼接成XML請求數(shù)據(jù)
StringBuilder sbPay = new StringBuilder();
foreach (KeyValuePair<string, string> k in sParams)
{
if (k.Key == "attach" || k.Key == "body" || k.Key == "sign")
{
sbPay.Append("<" + k.Key + "><![CDATA[" + k.Value + "]]></" + k.Key + ">");
}
else
{
sbPay.Append("<" + k.Key + ">" + k.Value + "</" + k.Key + ">");
}
}
return_string = string.Format("<xml>{0}</xml>", sbPay.ToString().TrimEnd(','));
return return_string;
}
public string PostXmlAndCertToUrl(string url, string postData,string cert,string password)
{
string resp = string.Empty;
ServicePointManager.ServerCertificateValidationCallback = new System.Net.Security.RemoteCertificateValidationCallback(CheckValidationResult);
//調(diào)用證書
System.Security.Cryptography.X509Certificates.X509Certificate2 cer = new System.Security.Cryptography.X509Certificates.X509Certificate2(cert, password, System.Security.Cryptography.X509Certificates.X509KeyStorageFlags.PersistKeySet | System.Security.Cryptography.X509Certificates.X509KeyStorageFlags.MachineKeySet);
HttpWebRequest webrequest = (HttpWebRequest)HttpWebRequest.Create(url);
webrequest.ClientCertificates.Add(cer);
webrequest.Method = "post";
webrequest.ContentType = "application/x-www-form-urlencoded";
webrequest.ContentLength = postData.Length;
//webrequest.ContentType = "text/xml";
//byte[] data = System.Text.Encoding.UTF8.GetBytes(postData);
//webrequest.ContentLength = data.Length;
HttpWebResponse response = null;
try
{
StreamWriter swRequestWriter = new StreamWriter(webrequest.GetRequestStream());
swRequestWriter.Write(postData);
if (swRequestWriter != null)
swRequestWriter.Close();
response = (HttpWebResponse)webrequest.GetResponse();
using (StreamReader reader = new StreamReader(response.GetResponseStream(), Encoding.UTF8))
{
resp = reader.ReadToEnd();
}
}
catch (Exception exp)
{
throw exp;
}
finally
{
if (response != null)
response.Close();
}
return resp;
}
public string getNoncestr()
{
Random random = new Random();
return GetMD5(random.Next(1000).ToString(), "GBK").ToLower().Replace("s", "S");
}退款功能實現(xiàn)
假設(shè)點擊退款按鈕事件
protected void Button_Click(object sender, EventArgs e)
{
string appId = “”; //服務(wù)號的appId
string paySignKey = “”; //申請的支付簽名KEY;
string mch_id = “”; //申請的支付商戶ID
string OrderID = ""; //支付訂單號
string OrderAmount = (Convert.ToInt32((float.Parse(Amount.Text) * 100))).ToString(); //訂單支付金額,Amount.Text 支付金額
string RefundOrderAmount = (Convert.ToInt32((float.Parse(Amount.Text) * 100))).ToString(); //退款金額(Amount.Text)這里表示全額退款
string RefundOrderID = Guid.NewGuid().ToString().Replace("-", ""); //生成退款訂單號
//創(chuàng)建退款訂單
RefundOrder order = new RefundOrder();
order.appid = appId;
order.mch_id = mch_id;
order.out_trade_no = OrderID;
order.nonce_str = tenpay.getNoncestr();
order.out_refund_no = RefundOrderID;
order.total_fee = int.Parse(OrderAmount);
order.refund_fee = int.Parse(RefundOrderAmount);
order.op_user_id = mch_id;
string cert = “d:\\apiclient_cert.p12"; //退款證書路徑
//私鑰(在安裝證書時設(shè)置)
string password =""; //證書密碼
//創(chuàng)建訂單明細(xì)類,調(diào)用getRefundOrderDetail方法進行退款
RefundOrderDetail orderdetail = getRefundOrderDetail(order, paySignKey, cert, password);
string rv = ("退款訂單號:" + RefundOrderID + "<br>");
try
{
rv += ("退款金額:" + (double.Parse(orderdetail.total_fee) / 100).ToString() + "<br>");
}
catch (Exception eee)
{
rv += ("退款金額:<br>");
}
rv += ("<b>交易狀態(tài): " + (orderdetail.result_code == "SUCCESS" ? "退款申請成功" : "退款申請失敗") + "(" + orderdetail.result_code + ")" + "</b><br>");
rv += ("可能的錯誤描述:" + orderdetail.err_code_des);
}對賬
退款申請成功后,僅為申請狀態(tài),需要通過查詢退款情況以確定是否完成,該功能可以在考生方進行實現(xiàn),考生可隨時查詢自己的對帳情況。
以下是參考代碼,該代碼可實現(xiàn)支付與退款的查詢:
protected void queryOrder(object sender, EventArgs e)
{
string OrderID =”“; //訂單號
string paytype = ”“; //查詢類型,支付消費或退款
string appId = ""; //服務(wù)號 appid
string paySignKey = ""; //支付簽名key
string mch_id = ""; //支付商戶號
if (paytype == "消費")
{
try
{
string openid = ”“;
QueryOrder order = new QueryOrder();
order.appid = appId;
order.mch_id = mch_id;
order.out_trade_no = OrderID;
order.nonce_str = getNoncestr();
OrderDetail orderdetail = getOrderDetail(order, paySignKey);
string rv = ("訂單號:" + OrderID + "<br>");
rv += ("付款人ID比對識別:" + (openid == orderdetail.openid ? "成功" : "失敗") + "<br>");
rv += ("交易金額:" + (double.Parse(orderdetail.total_fee) / 100).ToString() + "<br>");
rv += ("<b>交易狀態(tài): " + (orderdetail.trade_state == "SUCCESS" ? "成功" : "失敗") + "(" + orderdetail.trade_state + ")" + "</b><br>");
rv += ("支付交易時間:" + (orderdetail.time_end != "" && orderdetail.time_end.Length == 14 ? orderdetail.time_end.Substring(0, 4) + "-" + orderdetail.time_end.Substring(4, 2) + "-" + orderdetail.time_end.Substring(6, 2) + " " + orderdetail.time_end.Substring(8, 2) + ":" + orderdetail.time_end.Substring(10, 2) + ":" + orderdetail.time_end.Substring(12, 2) : "") + "<br>");
}
catch (Exception ex)
{
return;
}
}
else if (paytype == "退款")
{
try
{
RefundOrder order = new RefundOrder();
order.appid = appId;
order.mch_id = mch_id;
order.out_trade_no = OrderID;
order.nonce_str = getNoncestr();
RefundOrderDetail orderdetail = getRefundQueryOrderDetail(order, paySignKey);
string rv = ("<b>交易狀態(tài): " + (orderdetail.result_code == "SUCCESS" ? "成功" : "失敗") + "(" + orderdetail.result_code + ")" + "</b><br>");
rv += ("其它說明:" + orderdetail.err_code_des);
}
catch (Exception ex)
{
}
}
}支付商家后臺相關(guān)要點
實時交易帳單查詢
登錄后臺后,該操作可以進行實時交易的帳單對帳功能,以備在爭議的時候進行查詢,基本操作如下圖:

點擊交易中心、交易訂單、批量訂單查詢、查詢即可下載EXCEL格式的訂單。
精確交易帳單查詢
登錄后臺后,可查詢精確交易帳單,該帳單每天10:00更新前一天的數(shù)據(jù)交易,我們可以進行CSV格式的下載,操作如下圖:

點擊交易中心、交易帳單、打包下載即可,請注意圖中圈注的提示。
小結(jié)
文章提供的代碼僅供參考,在實際的應(yīng)用中,我們還可以根據(jù)業(yè)務(wù)需要編寫其它功能,如下載微信官方對帳單,導(dǎo)入到應(yīng)用系統(tǒng)中,與業(yè)務(wù)數(shù)據(jù)進行對帳,以排查爭議數(shù)據(jù);查詢訂單結(jié)果狀態(tài)以更新業(yè)務(wù)爭議狀態(tài)信息等。
以上就是C#實現(xiàn)微信退款及對賬功能的示例詳解的詳細(xì)內(nèi)容,更多關(guān)于C#實現(xiàn)微信退款及對賬的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Unity Shader實現(xiàn)動態(tài)過場切換圖片效果
這篇文章主要為大家詳細(xì)介紹了Unity Shader實現(xiàn)動態(tài)過場切換圖片效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-07-07
.Net WInform開發(fā)筆記(二)Winform程序運行結(jié)構(gòu)圖及TCP協(xié)議在Winform中的應(yīng)用
中午沒事,把去年剛畢業(yè)那會畫的幾張圖翻出來了,大概介紹Winform應(yīng)用程序運行的過程,以及TCP協(xié)議在Winform中的應(yīng)用。感興趣的朋友可以了解下;如果有Windows消息機制等基礎(chǔ),很好理解這兩張2013-01-01
C# IEnumerable和IEnumerator接口淺析
本文主要介紹了C#中IEnumerable和IEnumerator接口的相關(guān)知識,具有很好的參考價值,下面跟著小編一起來看下吧2017-02-02

