C# 实现微信退款及对帐

目录

需求

基础准备

关键代码

操作界面

​编辑

退款订单类及方法

退款功能实现

对帐

支付商家后台相关要点

实时交易帐单查询

精确交易帐单查询

小结

需求

在招聘报名系统里,考务费支付是其中一个环节,支付方式很多种,比如银联、微信、支付宝等等。本次我们以微信支付进行举例,在考生注册账号、编写简历、报名职位、被初审核通过等一系列基础的条件的具备下,可以进入支付考务费的环节(笔试费用),我们会为其生成一个支付二维码,考生支付后(无论成功与否),都会记录其支付结果状态。

在实际的应用中,对于支付成功的考生,我们会遇到实现退款的需求,只要包括如下场景:

1、根据政策规定,某些符合全部或部分退款条件的考生。

2、其它未知原因,重复支付订单的考生。

3、其它不可抗力,需求进行退款的考生。

基础准备

在实现功能前,做为企业,我们需要申请一个微信服务号,并成为微信支付商家。

1、申请服务号

申请成功后会获得到 AppId 和 AppSecret 用于后续开发,如关联支付商户、网页授权登录等。

具体指引请参照微信公众平台首页:https://mp.weixin.qq.com/cgi-bin/loginpage

2、成为微信支付商家

申请成功后会获得 Mchid 和 paySignKey 用于微信支付、退款等,请在商家后台务必关联申请的公众号。

具体指引请参照微信支付平台首页:https://pay.weixin.qq.com/index.php/core/home/login

上述两个平台申请成功后,请登录微信支付商家平台,进行如下图操作:

在产品中心、AppID帐号管理、关联 AppID(即申请的服务号) 

另外一个重要配置是支付目录,我们写的支付程序需要在这里设置,如下图:

关键代码

操作界面

界面上会显示最近一笔的微信订单支付情况,包括订单号、交费时间、交费金额、退款金额。其中退款金额不能大于成功交费金额,否则会返回失败。另外,还可以显示微信交易跟踪日志列表信息,如果订单号、交易价格、openid、返回信息、交易状态等。

示例界面如下:

退款订单类及方法

实现微信退款,需要在支付商家平台申请退款证书,证书文件保存到自定义的目录中,在退款时指定路径。

退款示例代码如下:

                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地址
//退款订单明细类
public class RefundOrderDetail{/// <summary>/// 返回状态码,SUCCESS/FAIL 此字段是通信标识,非交易标识,交易是否成功需要查看trade_state来判断/// </summary>public string return_code = "";/// <summary>/// 返回信息返回信息,如非空,为错误原因 签名失败 参数格式校验错误/// </summary>public string return_msg = "";/// <summary>/// 业务结果,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>/// 微信支付分配的终端设备号/// </summary>public string device_info = "";/// <summary>/// 随机字符串,不长于32位/// </summary>public string nonce_str = "";/// <summary>/// 签名/// </summary>public string sign = "";/// <summary>/// 微信支付订单号/// </summary>public string transaction_id = "";/// <summary>/// 商户系统的订单号,与请求一致。/// </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>/// 微信支付分配的终端设备号/// </summary>public string device_info = "";/// <summary>/// 随机字符串,不长于 32 位/// </summary>public string nonce_str = "";/// <summary>/// 签名public string sign = "";public string sign_type = "";/// <summary>/// 商户系统内部的订单号,32个字符内、可包含字母,确保在商户系统唯一,详细说明/// </summary>public string transaction_id = "";public string out_trade_no = "";public string out_refund_no = "";/// <summary>/// 订单总金额,单位为分,不能带小数点/// </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>/// 微信订单号,优先使用/// </summary>public string transaction_id = "";/// <summary>/// 商户系统内部订单号/// </summary>public string out_trade_no = "";/// <summary>/// 随机字符串,不长于 32 位/// </summary>public string nonce_str = "";/// <summary>/// 签名,参与签名参数:appid,mch_id,transaction_id,out_trade_no,nonce_str,key/// </summary>public string sign = "";}//申请退款方法,返回退款订单明细类
//参数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请求数据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);//调用证书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");}

退款功能实现

假设点击退款按钮事件

protected void Button_Click(object sender, EventArgs e)
{string appId = “”; //服务号的appIdstring paySignKey = “”; //申请的支付签名KEY;string mch_id = “”;  //申请的支付商户IDstring 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("-", "");   //生成退款订单号//创建退款订单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";  //退款证书路径//私钥(在安装证书时设置)string password =""; //证书密码
//创建订单明细类,调用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>交易状态:&nbsp;" + (orderdetail.result_code == "SUCCESS" ? "退款申请成功" : "退款申请失败") + "(" + orderdetail.result_code + ")" + "</b><br>");rv += ("可能的错误描述:" + orderdetail.err_code_des);
}

对帐

退款申请成功后,仅为申请状态,需要通过查询退款情况以确定是否完成,该功能可以在考生方进行实现,考生可随时查询自己的对帐情况。

以下是参考代码,该代码可实现支付与退款的查询:

protected void queryOrder(object sender, EventArgs e)
{string OrderID =”“; //订单号string paytype = ”“;  //查询类型,支付消费或退款string appId = "";   //服务号 appidstring paySignKey = "";   //支付签名keystring 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>交易状态:&nbsp;" + (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>交易状态:&nbsp;" + (orderdetail.result_code == "SUCCESS" ? "成功" : "失败") + "(" + orderdetail.result_code + ")" + "</b><br>");rv += ("其它说明:" + orderdetail.err_code_des);}catch (Exception ex){}}
}

支付商家后台相关要点

实时交易帐单查询

登录后台后,该操作可以进行实时交易的帐单对帐功能,以备在争议的时候进行查询,基本操作如下图:

点击交易中心、交易订单、批量订单查询、查询即可下载EXCEL格式的订单。

精确交易帐单查询

登录后台后,可查询精确交易帐单,该帐单每天10:00更新前一天的数据交易,我们可以进行CSV格式的下载,操作如下图:

点击交易中心、交易帐单、打包下载即可,请注意图中圈注的提示。 

小结

以上提供的代码仅供参考,在实际的应用中,我们还可以根据业务需要编写其它功能,如下载微信官方对帐单,导入到应用系统中,与业务数据进行对帐,以排查争议数据;查询订单结果状态以更新业务争议状态信息等。

以上就是自己的一些分享,时间仓促,不妥之处还请大家批评指正!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/177711.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

操作系统CLOCK算法

操作系统时钟(CLOCK)置换算法_时钟置换算法-CSDN博客 前七步相同 第八步的时候 发现页面在内存中 标记位变成1 但是指针不需要移动。

什么软件可以去视频水印?分享3个超实用去水印工具

什么软件可以去视频水印&#xff1f;短视频已然成为了我们日常生活或工作的一部分&#xff0c;当我们遇到感兴趣的视频想保存发现无法保存&#xff0c;或者保存后留有水印&#xff0c;非常影响我们视频观看度和分享欲&#xff0c;为了解决这一问题&#xff0c;许多针对视频水印…

如何用CHAT写“科技探索者”视频号运营方案

问CHAT&#xff1a;生成一篇“科技探索者”视频号运营方案&#xff0c;要求内容&#xff1a; &#xff08;1&#xff09;视频号的定位、面向的人群、主要发布哪方面的内容 &#xff08;2&#xff09;视频号的内容设计&#xff08;用什么样的方式来体现、最好有内容创意&#xf…

学习笔记-瑞吉外卖项目实战(一)

软件开发整体介绍 软件开发流程 角色分工 软件环境 瑞吉外卖项目介绍 项目介绍 产品原型介绍 技术选型 功能架构 角色 开发环境搭建 数据 创建database reggie&#xff0c;在里面创建表&#xff1a; maven 创建springboot项目并导入相关依赖坐标&#xff1a; 我们可以在项目…

【已解决】AttributeError: module ‘matplotlib‘ has no attribute ‘imshow‘

首先 在学习OpenCV的第一个程序&#xff0c;碰到这个问题记录一下。首先我已经安装好了matplotlib 如下图&#xff1a; 所以可以排除的就是我已经具备了这个库&#xff0c;那就是我在调用的时候出现的问题。 其次 回到pycharm&#xff0c;检查一下代码并做出如下的修改。 …

通达信抛物线SAR指标原理详解、参数设置及选股公式

抛物线指标(SAR)是由技术分析大师威尔斯威尔德(Welles Wilder)发明的&#xff0c;在其1978 年出版的《技术交易系统新概念》一书中介绍了该指标。SAR指标通过跟踪股票价格的动态变化&#xff0c;在走势图上以一系列点的形式显示&#xff0c;提供了一种判断趋势反转的方法&#…

鸿蒙应用开发-初见:入门知识、应用模型

基础知识 Stage模型应用程序包结构 开发并打包完成后的App的程序包结构如图 开发者通过DevEco Studio把应用程序编译为一个或者多个.hap后缀的文件&#xff0c;即HAP一个应用中的.hap文件合在一起称为一个Bundle&#xff0c;bundleName是应用的唯一标识 需要特别说明的是&…

某医院小程序存在支付漏洞和越权

某医院小程序存在支付漏洞和越权查看他人身份证&#xff0c;手机号&#xff0c;住址等信息 一个医院线上的小程序 登陆后点击个人信息&#xff0c;抓包&#xff0c;放到repeter模块&#xff0c; 修改strUserID参数可以越权查看别人信息 放intruder模块可以跑数据&#xff0c;这…

Redis缓存设计典型问题

目录 缓存穿透 缓存失效&#xff08;击穿&#xff09; 缓存雪崩 热点缓存key重建优化 缓存与数据库双写不一致 缓存穿透 缓存穿透是指查询一个根本不存在的数据&#xff0c; 缓存层和存储层都不会命中&#xff0c; 通常出于容错的考虑&#xff0c; 如果从存储层查不到数据…

MybatisPlus改造逻辑删除有多方便

MybatisPlus的逻辑删除可以有效保留历史数据。之前没有用逻辑删除的项目&#xff0c;想改造成逻辑删除总共需要几步&#xff1f; 答案&#xff1a;4步搞定 一、修改pom.xml的MybatisPlus版本&#xff08;注意版本兼容性&#xff09; <properties>...<!--<mybatis-…

anyRTC 融合音视频能力底座:助力企业数字化转型

随着全球化的发展&#xff0c;产业竞争日益激烈。数字化转型和创新成为了企业提高竞争力、实现可持续发展的重要手段&#xff0c;面对产业结构调整、资源环境挑战、数字技术与创新带来的行业颠覆与机遇&#xff0c;企业需要进行数字化转型和创新以适应新环境和新时代的挑战。 …

P13 C++ 类 | 结构体内部的静态static

目录 01 前言 02 类内部创建静态变量的例子 03 在类的内部创建静态变量的作用 04 最后的话 01 前言 本期我们讨论 static 在一个类或一个结构体中的具体情况。 在几乎所有面向对象的语言中&#xff0c;静态在一个类中意味着特定的东西。这意味着在类的所有实例中&#xff…

Springbot启动报错-类文件具有错误的版本 61.0, 应为 52.0

错误再现&#xff1a; 启动Springboot项目时候报错 java: 无法访问org.springframework.boot.SpringApplication 错误的类文件: /D:/Maven/apache-maven-3.6.3/repository/org/springframework/boot/spring-boot/3.0.0/spring-boot-3.0.0.jar!/org/springframework/boot/Sprin…

vue2+el-select实现分页加载更多功能

需求&#xff1a; 由于项目下拉框数据过多&#xff0c;一次性加载完&#xff0c;会有性能问题&#xff0c;于是希望可以增加分页加载更多功能。 实现效果&#xff1a; 如上图&#xff1a;点击“点击加载更多”按钮&#xff0c;实现分页加载下一页&#xff0c;直到最后一页&am…

香港优才计划是什么意思?一文详解2023年最新政策!

香港优才计划是什么意思&#xff1f;一文详解2023年最新政策&#xff01; 目前香港优才计划申请火热&#xff0c;但是还是有很多新手不太了解这个项目&#xff0c;跟风申请绝对不是什么好事&#xff0c;先了解清楚再考虑也是对自己对家人的一种交代。这篇文章就再来科普下。 优…

WEB渗透—反序列化(六)

Web渗透—反序列化 课程学习分享&#xff08;课程非本人制作&#xff0c;仅提供学习分享&#xff09; 靶场下载地址&#xff1a;GitHub - mcc0624/php_ser_Class: php反序列化靶场课程&#xff0c;基于课程制作的靶场 课程地址&#xff1a;PHP反序列化漏洞学习_哔哩哔_…

基于若依的ruoyi-nbcio流程管理系统增加流程节点配置(二)

更多ruoyi-nbcio功能请看演示系统 gitee源代码地址 前后端代码&#xff1a; https://gitee.com/nbacheng/ruoyi-nbcio 演示地址&#xff1a;RuoYi-Nbcio后台管理系统 上一节把数据库与相关基础数据字典准备好&#xff0c;下面就来实现相应的功能&#xff0c;目前先针对自定义…

实现了父类 纯虚函数为什么还有 无法解析外部符号错误

使用背景&#xff1a; 将C 的函数或接口使用 pybind11 封装成可以供python 使用调用的接口或函数&#xff0c;使用了CMake 编译&#xff08;若之前可以编译通过&#xff0c;现在编译不通过&#xff0c;重新选择 source code 路径&#xff09;成 VS 2019 可使用的目标解决方案&a…

ThinkPHP6学生选课管理系统

有需要请加文章底部Q哦 可远程调试 ThinkPHP6学生选课管理系统 一 介绍 此学生选课管理系统基于ThinkPHP6框架开发&#xff0c;数据库mysql8&#xff0c;前端bootstrap。系统角色分为学生&#xff0c;教师和管理员。学生登录后可进行选课&#xff0c;教师登录后可查看选课情况…

碧莲盛 x Tapdata:实时数据如何赋能医疗美容行业,助力医疗决策及个性化服务升级

使用 Tapdata&#xff0c;化繁为简&#xff0c;轻量代替 OGG、DSG 这样的同步工具&#xff0c;以及 Kettle、Informatica、Python 这样的 ETL 工具或脚本&#xff0c;帮助企业在五花八门的数据需求面前&#xff0c;实现“做且仅做最后一次 ETL”的目标&#xff0c;这绝非纸上谈…