当前位置: 首页 > 编程日记 > 正文

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 = “”; //服务号的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("-", "");   //生成退款订单号


//创建退款订单
        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 = "";   //服务号 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>交易状态:&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格式的下载,操作如下图:

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

小结

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

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

相关文章:

并发编程下的集合:数组寻址、LinkedList、HashMap、ConcurrentHashMap

如果发现hash取模后的数组索引位下无元素则直接新增,若不是空那就说明存在hash冲突,则判断数组索引位链表结构中的第一个元素的key以及hash值是否与新的key一致则直接覆盖,若不一致则判断当前的数组索引下的链表结构是否为红黑树,若为红黑树则走红黑树的新增方法,若不为红黑树则遍历当前链表结构,遍历中发现某个节点元素的next为null是则直接将新元素指针与next进行关联,若在遍历到next为空前判断到,某个节点的key以及key的hash值与新的key与新的keyhash值一致时则走覆盖。

【日常开发之Windows共享文件】Java实现Windows共享文件上传下载

下拉框选择你选择的用户点击添加,然后共享确定。创建一个文件夹然后点击属性界面,点击共享。maven版本存在于SMB协议的兼容问题。首先开启服务,打开控制面板点击程序。点击启用或关闭Windows功能。我这边是专门创建了一个用户。SMB1.0选中红框内的。

rust wasm入门

demo## 编译 Rust 为 WebAssembly在本教程中,我们将使用 Rust 的 npm 包构建工具 wasm-pack 来构建一个 npm 包。

Kubernetes对象的定义和操作

Kubernetes对象指的是Kubernetes系统的持久化实体,所有这些对象合起来,代表了你集群的实际情况。常规的应用里,我们把应用程序的数据存储在数据库中,Kubernetes将其数据以Kubernetes对象的形式通过 api server存储在 etcd 中。集群中运行了哪些容器化应用程序集群中对应用程序可用的资源应用程序相关的策略定义,例如,重启策略、升级策略、容错策略其他Kubernetes管理应用程序时所需要的信息。

使用名称空间共享集群

可以限定使用某个名称空间的用户不能看到另外一个名称空间中的内容。默认情况下,安装Kubernetes集群时,会初始化一个 default 名称空间,用来将承载那些未指定名称空间的 Pod、Service、Deployment等对象。接下来,为 kubectl 定义一个上下文,以便在不同的名称空间中工作。此时,开发人员可以做任何他想要做的操作,所有操作都限定在名称空间 development 里,而无需担心影响到 production 名称空间中的内容。使用 kubectl 有两种方式可以创建名称空间。

k8s 标签和选择器

标签(Label)是附加在Kubernetes对象上的一组名值对,其意图是按照对用户有意义的方式来标识Kubernetes对象,同时,又不对Kubernetes的核心逻辑产生影响。管理这些对象时,很多时候要针对某一个维度的条件做整体操作,例如,将某个版本的程序整体删除,这种情况下,如果用户能够事先规划好标签的使用,再通过标签进行选择,就会非常地便捷。Kubernetes api server支持两种形式的标签选择器,equality-based 基于等式的 和 set-based 基于集合的。

iced 入门一

本教程的目标是创建一个简单的购物清单应用程序。我们希望允许添加和删除购物清单中的项目。在编写代码之前,我们必须首先了解 Iced 构建的结构:Elm 架构。它是 GUI 库使用的架构,最初用于 Elm 编程语言。它的核心原则很简单。它围绕三个概念构建:模型、视图和更新。

注解annotation

Kubernetes的系统组件(例如,kube-scheduler、kube-controller-manager、kube-apiserver、kubectl 或其他第三方组件)向用户的Kubernetes对象添加注解时,必须指定一个前缀。注解(annotation)可以用来向 Kubernetes 对象的 metadata.annotations 字段添加任意的信息。除了使用注解,您也可以将这类信息存放在一个外部的数据库,然而,在使用、分享这些信息的时候,可能会变得难以管理。

Deployment概述

Pod 容器组是 Kubernetes 中最小的调度单元,更多信息请参考 容器组 - 概述Deployment 是最常用的用于部署无状态服务的方式。Deployment 控制器使得您能够以声明的方式更新 Pod(容器组)和 ReplicaSet(副本集)。以“声明”的方式管理 Pod 和 ReplicaSet,其本质是将一些特定场景的一系列运维步骤固化下来,以便快速准确无误的执行。

Rust XTask 模式介绍与应用

XTask(扩展任务)是一种在Rust项目中定义和执行自定义构建任务的方式。它通过创建一个独立的Rust库或二进制项目来封装这些任务,利用Rust语言的强类型、安全性和跨平台能力,使得构建流程更加健壮、可读和可维护。

ModuleNotFoundError: No module named ‘qcloud_cos‘

是腾讯云提供的一个Python SDK,用于与腾讯云对象存储(COS)服务进行交互。使用pip安装qcloud_cos报以下错误。这个错误表示Python无法找到名为。

需要在method方法被调用之后,仅打印出a=100,b=200,请写出method方法的代码

通常,此流对应于显示器输出或者由主机环境或用户指定的另一个输出目标。通常,此流对应于键盘输入或者由主机环境或用户指定的另一个输入源。public static final PrintStream err“标准”错误输出流。PrintStream 是打印输出流,它继承于FilterOutputStream。第二个用的是用的是char类型,根本不是方法,当要输出方法体的时候,会给你遍历数组。通常,此流对应于显示器输出或者由主机环境或用户指定的另一个输出目标。诡异的是,如果错了,面试官对你说了一句:你回去看看,

一个合格的Java选手必须要掌握的并发锁知识

Java内置锁:基于Java语法层面(关键词)实现的锁,主要是根据Java语义来实现,最典型的应用就是synchronized。Java显式锁:基于JDK层面实现的锁,主要是根据基于Lock接口和ReadWriteLock接口,以及统一的AQS基础同步器等来实现,最典型的有ReentrantLock。使用方式:synchronized关键字互斥锁主要有作用于对象方法上面,作用于类静态方法上面,作用于对象方法里面,作用于类静态方法里面等4种方式。

Integer.toHexString(b & 0xff)理解以及& 0xff什么意思

首先toHexString传的参数应该是int类型32位,此处传的是byte类型8位,所以前面需要补24个0。然后& 0xff 就是把前面24个0去掉只要后8位。toHexString(b & 0xff)相当于做了一次位的与运算,将前24位字符省略,将后8位保留。是两个十六进制的数,每个f用二进制表示是1111,所以占四位(bit),两个f()占八位(bit),八位(bit)也就是一个字节(byte).这个方法是把字节(转换成了int)以16进制的方式显示。我的理解是这样,如有不对欢迎指正!

k8s搭建部署(超详细)

Kubernetes是Google 2014年创建管理的,是Google 10多年大规模容器管理技术Borg的开源版本。它是容器集群管理系统,是一个开源的平台,可以实现容器集群的自动化部署、自动扩缩容、维护等功能。快速部署应用快速扩展应用无缝对接新的应用功能节省资源,优化硬件资源的使用可移植: 支持公有云,私有云,混合云,多重云(multi-cloud)可扩展: 模块化, 插件化, 可挂载, 可组合自动化: 自动部署,自动重启,自动复制,自动伸缩/扩展。

【OpenCV】在Linux上使用OpenCvSharp

OpenCvSharp是一个OpenCV的 .Net wrapper,应用最新的OpenCV库开发,使用习惯比EmguCV更接近原始的OpenCV,该库采用LGPL发行,对商业应用友好。

使用JavaScript实现复杂功能:一个完整的电商网站搜索功能

随着互联网的发展,电子商务网站已经成为人们购物的重要平台。而在这些网站中,搜索功能无疑是核心功能之一。用户可以通过搜索快速找到他们需要的商品,从而提高购物体验。本文将详细介绍如何使用JavaScript实现一个完整的电商网站搜索功能。

C++并发编程:互斥锁std::mutex和lock_guard的使用

对象离开其作用域时,会自动调用析构函数,该析构函数会释放锁。这确保了在任何情况下(包括由于异常等原因导致的提前退出),锁都会被正确释放,从而避免了忘记手动释放锁而导致的死锁问题。mutex 用于控制多个线程访问共享资源,确保在任意时刻只有一个线程可以访问该资源,避免数据竞争。这确保了同一时刻只有一个线程可以访问被保护的资源,从而防止多线程并发访问导致的数据不一致性。是 C++ 标准库中提供的一个模板类,用于在其构造时自动获取锁,在析构时自动释放锁。是 C++ 标准库中提供的一种用于多线程同步的互斥锁实现。

上位机图像处理和嵌入式模块部署(qt插件的使用)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】 一个软件一般有很多的功能,但是主流程只有一个。但在软件开发的过程当中,一般来说功能是需要不断添加的,但是主流程最好不要轻易修改。这里的插件就相当于各种各样的功能,而主流程就是如何怎么去调用这些插件的功能。所以,今天正好来学一下怎么添加qt插件,个人觉得这部分还是非常重要的。

k8s图形化管理工具之rancher

在前面的k8s基础学习中,我们学习了各种资源的搭配运用,以及命令行,声明式文件创建。这些都是为了k8s管理员体会k8s的框架,内容基础。在真正的生产环境中,大部分的公司还是会选用图形化管理工具来管理k8s集群,大大提高工作效率。在二进制搭建k8集群时,我们就知道了k8s本身就具有一款原生的k8s集群管理工具,但是原生图形化管理工具dashborad只拥有管理一个集群的能力。而对于现代化生产力公司来讲,一个集群能够做的事情还是太少,所以我们需要引入更强大的集群管理工具。

C程序的内存空间布局(栈、堆、数据区、常量区、代码区)

较详细的介绍了栈、堆、数据区、常量区、代码区

给服务器开通telnet的流程

但一些特殊场景下,比如要升级ssh,ssh不能用时,需要使用telnet,用过要关闭此服务。需要首先安装,如果telnet-server服务在xinetd之前安装了,要先删除telnet-server,再安装xinetd。安装顺序:xinetd--》telnet--》telnet-server。安装顺序:xinetd--》telnet--》telnet-server。2、卸载rpm包(如果已经安装了,又不清楚顺序,可以都卸载后统一安装)注意:telnet-server服务启动依赖xinetd服务,

Java中的四种访问权限(private,public,protected,无修饰)

/实体类属性和数据库字段名称不一致//实体类属性和数据库字段名称不一致return id;return age;emp.test();//直接调用public修饰的变量//private修饰的变量进行赋值//调用private修饰的变量1、public修饰符定义的属性和方法通过对象实例化进行调用,2、private修饰的属性通过set、get方法进行调用。

微信小程序之WXSS模板样式、页面配置(.json)和网络数据请求

一、WXSS 模板样式1、什么是 WXSS2、WXSS 和 CSS 的关系二、WXSS 模板样式 - rpx1、什么是 rpx 尺寸单位2、rpx 的实现原理3、rpx 与 px 之间的单位换算*三 、WXSS 模板样式 - 样式导入1、什么是样式导入2、@import 的语法格式四、WXSS 模板样式 - 全局样式和局部样式1、全局样式2、局部样式五、页面配置1、页面配置文件的作用2、页面配置和全局配置的关系3、页面配置中常用的配置项。

Java中的方法重载和方法重写有什么区别?

Java中的方法重载(Overloading)和方法重写(Overriding)都是面向对象编程中的重要概念,但它们之间有一些区别。方法重载是指在同一个类中,可以定义多个具有相同名称但参数列表不同的方法。这些方法具有不同的参数类型、参数个数或参数顺序。在调用重载方法时,Java编译器会根据传递给方法的参数类型和数量来选择要调用的正确方法。方法重载主要用于解决方法的命名冲突和提高代码的可读性和可维护性。

python基础使用之变量,表达式,语句

PYTHON基础知识系列之变量、表达式、语句

C语言常见面试题:什么是宏,宏的作用是什么?

宏在计算机科学中是一种批量处理程序命令,它是一种抽象的规则或模式,用于说明某一特定输入(通常是字符串)如何根据预定义的规则转换成对应的输出(通常也是字符串)。在编译时,预处理器会对宏进行展开,即将宏的内容替换到宏所在的位置。以上是宏的一些主要作用,但并不是全部。在实际编程中,根据需要选择是否使用宏以及如何使用宏,以实现更好的代码组织和可读性。,这样就可以计算出a和b的和。这个例子展示了宏的基本用法和作用。在这个例子中,我们定义了一个宏。,用于计算两个数的和。时,预处理器会将其展开为。

python基础小知识:引用和赋值的区别

通过引用,就可以在程序范围内任何地方传递大型对象而不必在途中进行开销巨大的赋值操作。不过需要注意的是,这种赋值仅能做到顶层赋值,如果出现嵌套的情况下仍不能进行深层赋值。赋值与引用不同,复制后会产生一个新的对象,原对象修改后不会影响到新的对象。如果在原位置修改这个可变对象时,可能会影响程序其他位置对这个对象的引用

k8s 使用tomcat官方镜像部署集群并解决访问页面404

官方镜像这里有个坑,使用kubectl启动之后,页面报错404,仔细检查发现,是因为tomcat的webapp目录下没有对应的文件,所以连初始界面都无法显示。要想显示,必须要根据官方镜像自己构建一个Dockerfile。根据上面的信息可以看出,该POD部署在k8s-node1上,映射POD的8080端口到master的30088端口上。这里需要将镜像上传到自己搭建的registry,并配置nodes节点都可以正常访问5000端口。三、根据官方镜像自己构建一个一次性就能启动的Tomcat镜像。

在C#中调用C++函数并返回const char*类型的值

在C#中,使用DllImport特性将C++函数声明为外部函数。在Main方法中,调用generateProjectCode函数并将返回的指针转换为const char*类型的字符串。在C#中调用C++函数并返回const char*类型的值,可以使用Interop服务来实现。C++代码需要编译为动态链接库(DLL)。