Visual Studio 2019 C# 上位机入门(2):写一个简单的串口助手

news2024/11/20 9:10:46

前言

本文记录一下用Visual Studio 2019 C# 写一个简单的串口助手的过程,由于没有先从小处学习,而是直接找相关资料就开始做,免不了很多奇怪的问题花了一些时间,基于此情况,我将尽可能整理出更多细节,尤其是我遇到的坑,以便和我一样的新手小白上手。后续我还准备单独分析Visual Studio上提供的一些控件的使用方法,以加深理解。话不多说,正文开始

先看成品:
在这里插入图片描述

制作过程

1.创建项目,及工程(我用的是Visual Studio 2019)选择windows 窗体应用;前面创建项目这几步有疑问可以参考:Visual Studio 2019 C# 上位机入门(1)
在这里插入图片描述
在这里插入图片描述
2. 设置项目名、保存路径、还有框架,最后点击创建;
在这里插入图片描述
3.创建项目完成后,进入工程管理器,左边白色部分就是Form编辑窗口;
在这里插入图片描述
4. 点击打开视图=》工具箱,左边栏就出现了,各控件列表,这就是我们做上位机最基本的部分,点开公共控件、容器等就可以看到我们常用到的各个组件了;
在这里插入图片描述
在这里插入图片描述
5.从工具箱中拖出一个控件比如Button,用鼠标点击选中,界面右下角则显示出其属性;
在这里插入图片描述

6.现在添加我们需要的控件,从工具箱中

先点击Form窗体,右下角出来的属性中找到Text改成串口调试助手,背景色改成喜欢的颜色
在这里插入图片描述
在这里插入图片描述

拖出5个GroupBox参考后面的成品图放到大概位置,调整大小,点击每个GroupBox分别将属性中的Text改为串口设置、定时发送、发送数据、接收数据,还有一个的Text填空格。GroupBox 控件是将其它的控件放入其中,形成一个控件组。

拖出6个Label参考后面的成品图放到大概位置,分别将其属性的Text设置为: 串口号:、波特率:、数据位:、停止位:、奇偶校验: 、时间:;

拖出5个ComboBox参考后面的成品图放到大概位置,分别将name(属性窗口往上拉就可以找到)设置为cbxCOMPort(对应串口号)、cbxBaudRate(对应波特率)、cbxDataBits(对应数据位)、cbxStopBits(对应停止位)、cbxParity(对应奇偶校验);

拖出2个RadioButton参考后面的成品图放到大概位置,属性中Text分别设置为:字符显示、HEX显示
name分别设置为rbnChar(对应字符显示)、rbnHex(对应HEX显示)Checked属性分别设置为:True、False(将这2个放在同一个GroupBox中,RadioButton控件只能被选中一个)

拖出4个Button参考后面的成品图放到大概位置,属性中Text分别设置为检测串口、打开串口、发送数据、清除数据,name分别设置为btnCheckCom(对应检测串口)、btnOpenCom(对应打开串口)、btnSendData(对应发送数据)、btnClearData(对应清除数据),注意对应关系千万别弄错

拖出3个CheckBox,Text分别设置为HEX发送、回车换行、自动发送;
自动发送的CheckBox的Name改为timeBox3,另外两个可以维持默认的Name

拖出一个NumericUpDown,放到“时间:”这个Label后面;

拖出2个TextBox,name分别设置为tbxRecvData、tbxSendData,接收的ReadOnly属性设置为True。
点击框上面的黑色小三角,选中MultiLine可任意伸缩大小。属性中的Font可修改字体大小等,ScrollBars设置为Both可垂直、水平拉伸信息;
BackColor设置成黑色,字体颜色设置成白色
在这里插入图片描述
在这里插入图片描述

拖出一个StatusStrip,放在窗体的底部位置,调整熟悉中背景色BackColor跟窗体一致,点击图中加号添加一个StatusLabel
在这里插入图片描述

拖出2个Timer放到窗体下面有个空白处;
在这里插入图片描述

点击Form1面板上边沿部分,选中面板,在属性中BackColor可修改背景颜色,AcceptButton设置为btnSendData,在窗体上回车关联到发送数据按钮,用户每次按“Enter”键都相当于按此按钮。Text可修改左上方名称,Icon可设置图标;
在这里插入图片描述
最终如图:
在这里插入图片描述
7.双击某个控件可跳转到对应的事件程序中,外框架搭好了,接下来就是写程序,完善其功能。
双击Form1面板上边沿部分进入Form1_Load中开始写程序
在Form1.cs中添加

using System.IO.Ports;
using System.Text.RegularExpressions;//正则表达式,加入命名空间。

程序源码:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

using System.IO.Ports;
using System.Text.RegularExpressions;//正则表达式,加入命名空间。

namespace 串口助手
{
    public partial class Form1 : Form
    {
        SerialPort sp = null;
        bool isOpen = false;
        bool isSetProperty = false;
        public Form1()
        {
            InitializeComponent();
        }
        private void Form1_Load(object sender, EventArgs e)
        {
            this.MaximizeBox = false;
            this.MaximumSize = this.Size;
            this.MinimumSize = this.Size;

            for (int i = 0; i < 100; i++)
            {
                cbxCOMPort.Items.Add("COM" + (i + 1).ToString());
            }
            cbxCOMPort.SelectedIndex = 0;

            cbxBaudRate.Items.Add("1200");
            cbxBaudRate.Items.Add("2400");
            cbxBaudRate.Items.Add("4800");
            cbxBaudRate.Items.Add("9600");
            cbxBaudRate.Items.Add("19200");
            cbxBaudRate.Items.Add("38400");
            cbxBaudRate.Items.Add("115200");
            cbxBaudRate.SelectedIndex = 6;

            cbxStopBits.Items.Add("0");
            cbxStopBits.Items.Add("1");
            cbxStopBits.Items.Add("1.5");
            cbxStopBits.Items.Add("2");
            cbxStopBits.SelectedIndex = 1;

            cbxParity.Items.Add("无");
            cbxParity.Items.Add("奇校验");
            cbxParity.Items.Add("偶校验");
            cbxParity.SelectedIndex = 0;

            cbxDataBits.Items.Add("8");
            cbxDataBits.Items.Add("7");
            cbxDataBits.Items.Add("6");
            cbxDataBits.Items.Add("5");
            cbxDataBits.SelectedIndex = 0;

            rbnChar.Checked = true;
            /*添加时间显示*/
            this.toolStripStatusLabel1.Text = "当前时间" + DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss");
            timer1.Interval = 1000;
            timer1.Start();
        }
        private void btnCheckCom_Click(object sender, EventArgs e)
        {
            bool comExistence = false;
            cbxCOMPort.Items.Clear();
            for (int i = 0; i < 100; i++)
            {
                try
                {
                    SerialPort sp = new SerialPort("COM" + (i + 1).ToString());
                    sp.Open();
                    sp.Close();
                    cbxCOMPort.Items.Add("COM" + (i + 1).ToString());
                    comExistence = true;
                }

                catch (Exception)
                {
                    continue;
                }
            }
            if (comExistence)
            {
                cbxCOMPort.SelectedIndex = 0;
            }
            else
            {
                MessageBox.Show("没有找到可用串口!", "错误提示!");
            }
        }
        private bool CheckPortSetting()
        {
            if (cbxCOMPort.Text.Trim() == "") return false;
            if (cbxBaudRate.Text.Trim() == "") return false;
            if (cbxStopBits.Text.Trim() == "") return false;
            if (cbxParity.Text.Trim() == "") return false;
            if (cbxDataBits.Text.Trim() == "") return false;
            return true;
        }
        private bool CheckSendData()
        {
            if (tbxSendData.Text.Trim() == "") return false;
            return true;
        }
        private void SetProperty()
        {
            sp = new SerialPort();
            sp.PortName = cbxCOMPort.Text.Trim();
            sp.BaudRate = Convert.ToInt32(cbxBaudRate.Text.Trim());
            if (cbxStopBits.Text.Trim() == "0")
            {
                sp.StopBits = StopBits.None;
            }
            else if (cbxStopBits.Text.Trim() == "1.5")
            {
                sp.StopBits = StopBits.OnePointFive;
            }
            else if (cbxStopBits.Text.Trim() == "2")
            {
                sp.StopBits = StopBits.Two;
            }
            else
            {
                sp.StopBits = StopBits.One;
            }

            sp.DataBits = Convert.ToInt16(cbxDataBits.Text.Trim());

            if (cbxParity.Text.Trim() == "奇校验")
            {
                sp.Parity = Parity.Odd;

            }
            else if (cbxParity.Text.Trim() == "偶校验")
            {
                sp.Parity = Parity.Even;
            }
            else
            {
                sp.Parity = Parity.None;
            }
            sp.ReadTimeout = -1;
            sp.RtsEnable = true;

            sp.DataReceived += new SerialDataReceivedEventHandler(sp_DataReceived);
        }
        private void sp_DataReceived(object sender, SerialDataReceivedEventArgs eg)
        {

            System.Threading.Thread.Sleep(100);

            this.Invoke((EventHandler)delegate//异步执行 一个线程
            {
                if (!rbnHex.Checked)//如果未选中name为rbnHex的控件
                {
                    //tbxRecvData.Text += sp.ReadLine();
                    StringBuilder sb = new StringBuilder();
                    long rec_count = 0;
                    int num = sp.BytesToRead;
                    byte[] recbuf = new byte[num];
                    rec_count += num;

                    sp.Read(recbuf, 0, num);
                    sb.Clear();

                    try
                    {
                        Invoke((EventHandler)(delegate
                        {
                            sb.Append(Encoding.ASCII.GetString(recbuf));  //将整个数组解码为ASCII数组
                            tbxRecvData.AppendText(sb.ToString());
                        }
                        )
                        );
                    }

                    catch
                    {
                        MessageBox.Show("请勾选换行", "错误提示");
                    }
                }
                else if (rbnHex.Checked)//如果选中
                {
                    Byte[] ReceivedData = new Byte[sp.BytesToRead];
                    sp.Read(ReceivedData, 0, ReceivedData.Length);

                    String RecvDataText = null;

                    for (int i = 0; i < ReceivedData.Length; i++)
                    {
                        RecvDataText += (ReceivedData[i].ToString("X2") + " ");//数组里接收到的数据转化为16进制
                    }
                    tbxRecvData.Text += RecvDataText;
                }
                sp.DiscardInBuffer();
            });
        }
        private void btnOpenCom_Click(object sender, EventArgs e)
        {
            if (isOpen == false)
            {
                if (!CheckPortSetting())
                {
                    MessageBox.Show("串口未设置", "错误提示");
                    return;
                }
                if (!isSetProperty)
                {
                    SetProperty();
                    isSetProperty = true;
                }
                try
                {
                    sp.Open();
                    isOpen = true;
                    btnOpenCom.Text = "关闭串口";
                    cbxCOMPort.Enabled = false;
                    cbxBaudRate.Enabled = false;
                    cbxDataBits.Enabled = false;
                    cbxParity.Enabled = false;
                    cbxStopBits.Enabled = false;
                    rbnChar.Enabled = false;
                    rbnHex.Enabled = false;
                }
                catch (Exception)
                {
                    isSetProperty = false;
                    isOpen = false;
                    MessageBox.Show("串口无效或已被占用", "错误提示");
                }
            }
            else if (isOpen == true)
            {

                try
                {
                    if (!timeBox3.Checked)
                    {
                        sp.Close();//关闭端口
                        isOpen = false;
                        btnOpenCom.Text = "打开串口";
                        cbxCOMPort.Enabled = true;
                        cbxBaudRate.Enabled = true;
                        cbxDataBits.Enabled = true;
                        cbxParity.Enabled = true;
                        cbxStopBits.Enabled = true;
                        rbnChar.Enabled = true;
                        rbnHex.Enabled = true;
                    }
                    else
                    {
                        MessageBox.Show("请先关闭自动发送", "错误提示");
                    }
                }
                catch (Exception)
                {
                    MessageBox.Show("关闭串口时发生错误", "错误提示");
                }
            }
        }
        private void btnSendData_Click(object sender, EventArgs e)
        {
            byte[] textchar = new byte[1];
            int num2 = 0;
            if (isOpen)
            {
                try
                {
                    if (!checkBox1.Checked)//如果没有选中十六进制发送
                    {
                        if (!checkBox2.Checked)//未选中回车换行
                        {
                            sp.Write(tbxSendData.Text);//串口发送 (发送框里的东西)
                        }
                        else
                        {
                            //sp.WriteLine(tbxSendData.Text);//用这个方法只会在后面加\n没有\r,下面的方法是验证可行的
                            string res = tbxSendData.Text + "\r\n";
                            byte[] byteArray = System.Text.Encoding.Default.GetBytes(res);
                            sp.Write(byteArray, 0, byteArray.Length);
                        }
                    }
                    else//选择十六进制发送的时候
                    {

                        string buf = tbxSendData.Text;
                        string bartenm = @"\s";//正则表达式
                        string replace = "";

                        Regex rgx = new Regex(bartenm);
                        string senddata = rgx.Replace(buf, replace);
                        num2 = (senddata.Length - senddata.Length % 2) / 2;

                        for (int a = 0; a < num2; a++)
                        {
                            textchar[0] = Convert.ToByte(senddata.Substring(a * 2, 2), 16);
                            sp.Write(textchar, 0, 1);
                        }


                        if (senddata.Length % 2 != 0)
                        {
                            textchar[0] = Convert.ToByte(senddata.Substring(tbxSendData.Text.Length - 1, 2), 16);
                            sp.Write(textchar, 0, 1);
                            num2++;
                        }
                    }
                }
                catch
                {
                    MessageBox.Show("发送数据时发生错误!", "错误提示");
                    return;
                }
            }
            else
            {
                MessageBox.Show("串口未打开错误提示!", "错误提示");
            }
            if (!CheckSendData())
            {
                MessageBox.Show("请输入要发送的数据", "错误提示");
            }
        }
        private void btnClearData_Click(object sender, EventArgs e)
        {
            if (!timeBox3.Checked)
            {
                tbxRecvData.Text = "";
                tbxSendData.Text = "";
            }
            else
            {
                MessageBox.Show("请先关闭自动发送", "错误提示");
            }
        }
        private void timer1_Tick(object sender, EventArgs e)
        {
            this.toolStripStatusLabel1.Text = "当前时间" + DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss");
        }
        private void timeBox3_CheckedChanged(object sender, EventArgs e)
        {
            System.Windows.Forms.Timer txTimer = new System.Windows.Forms.Timer();

            if (timeBox3.Checked)
            {
                if (numericUpDown1.Value != 0)
                {
                    if (CheckSendData())
                    {
                        txTimer.Enabled = false;
                        timer2.Interval = (int)numericUpDown1.Value; //定时器赋初值  
                        timer2.Start();
                    }
                    else if (!CheckSendData())
                    {
                        timer2.Stop();
                    }
                }
                else if (numericUpDown1.Value == 0)
                {
                    timer2.Stop();
                }
            }
            else
            {
                txTimer.Enabled = true;
                timer2.Stop();
            }
        }
        private void timer2_Tick(object sender, EventArgs e)
        {
            btnSendData_Click(btnSendData, new EventArgs());
        }
    }
}

8.最后是很重要的一步,刚才我们已经把程序全部编辑好了,如果是直接把我提供的程序复制粘贴的话,那么还需要做一个工作是回到窗体设计页面把每个控件中对应的触发事件设置对应我们编辑的成员函数才行。因为正常的开发应该是放置好控件后,单击控件,在右下角属性栏中点击小闪电,快速跳转到事件选项,双击某个事件就会跳到程序编辑界面,系统自动为你添加一个默认名称的事件,然后编辑这个事件。
在这里插入图片描述
那么现在我们是已经先编辑完了程序,然后再去在窗体设计界面把对应控件的触发事件设置一下对应哪个已经编辑好的函数就行了。

点击检测串口按钮,设置Click对应btnCheckCom_Click
在这里插入图片描述
点击打开串口按钮,设置Click对应btnOpenCom_Click
在这里插入图片描述
点击清空数据按钮,设置Click对应btnClearData_Click
在这里插入图片描述
点击发送数据按钮,设置Click对应btnSendData_Click
在这里插入图片描述
点击time1控件,设置Tick对应timer1_Tick,同理,点击time2控件,设置Tick对应timer2_Tick
在这里插入图片描述

9.最后点击顶部菜单栏的启动就可以开始调试验证了
在这里插入图片描述
找一个USB转串口模块,其TX和RX对插,连上电脑,点击检测串口,选择端口,点击打开串口,发送数据栏输入数据,点击发送数据,看接收数据是否正常(由于tx和rx对插,这样相当于自发自收,接收数据栏应该也会收到和发送一样的数据)。其他功能自行验证。
在这里插入图片描述
10.将Debug改为Release,点击生成解决方案
在这里插入图片描述
工程路径下的Release文件夹就生成了可执行文件,这个文件就可在其他电脑上使用了
在这里插入图片描述

如果读者朋友在练习实践的过程中遇到问题可能是我的过程写得不够细导致结果出现偏差,可以在评论中反馈出来以便改进文章质量。

这个串口助手还有很多功能可以添加,比如保存LOG文件功能,发送和接收字节计数,时间戳功能等等,后面有空再研究。

参考
https://blog.csdn.net/Casey_shi/article/details/117965159

https://www.freesion.com/article/8653228957/

https://blog.csdn.net/weixin_42378319/article/details/118424816

https://blog.csdn.net/C_gyl/article/details/78134311

https://www.haolizi.net/example/view_28754.html

https://www.feiqueyun.cn/zixun/jishu/115812.html

https://blog.csdn.net/You_are_blind/article/details/127010206

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

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

相关文章

Linux基础内容(20)—— 共享内存

Linux基础内容&#xff08;19&#xff09;—— 进程间通信(介绍与管道内容)_哈里沃克的博客-CSDN博客 目录 1.共享内存的原理 2.共享内存的概念和特点 创建共享内存 共享内存的形式 共享内存(ipc资源)的调用和特征 用户接口删除共享内存 共享内存关联 去关联 特点 …

React基础学习(一)

一、虚拟DOM和真实DOM <script type"text/babel"> // 此处一定要写babel!!!!!!!// 1. 创建虚拟DOM// const VDOM <h1 id"title">Hello, React!</h1> // 此处一定不要写引号 因为这不是字符串!!!!!!!const VDOM ( // 如果有多层嵌套&a…

PostgreSQL15.2最新版本安装_远程连接_Navicat操作_pgAdmin操作_Windows10上安装---PostgreSQL工作笔记001

首先去下载postgresql https://www.enterprisedb.com/downloads/postgres-postgresql-downloads 下载地址: 去上面的地址下载,最好下载10版本的,我这里下载的是15版本的,有问题,后面说吧 下载以后 然后双击安装 next 选择目录next next 输入密码next

好程序员:Java线下培训有必要吗?零基础想学Java怎么学?

有粉丝问好程序员&#xff1a;自己只有周末有时间&#xff0c;想报班学习Java编程&#xff0c;是线上学编程好还是线下学编程好&#xff1f;小源从实际客观以及学习效果的角度来讲&#xff0c;毫无疑问是线下学编程的效果会更好。为什么这样说呢&#xff1f; 比如&#xff1a;家…

初探强化学习

1.引言 人生中充满选择&#xff0c;每次选择就是一次决策&#xff0c;我们正是从一次次决策中&#xff0c;把自己带领到人生的下一段旅程中。在回忆往事的时候&#xff0c;我们会对生命中某些时刻的决策印象深刻&#xff1a;“还好当时选择了读研&#xff0c;毕业后找到了一份自…

学习小程序基础内容之逻辑交互

我们先来看一下实现的效果。 然后再来分享结构。 结构分为左右3:7 分配&#xff0c; 左侧是类别&#xff0c;右侧是该类别对应的品牌。 后台会在onload的请求把左侧的类别返回来&#xff0c;然后我们通过循环把数据展示出来。然后通过点击事件&#xff0c;把对应的品牌请求回来…

2023年,初级测试到高级测试开发工程师需要什么技能?卷起来......

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 Python自动化测试&…

泵站水闸自动化系统调试方案

1、调试方案概述 调试方案分为自控系统现场调试、视频系统现场调试以及控制中心远程调试&#xff08;包含通讯测试、自控调试和视频调试&#xff09;。 自控系统现场调试分为触摸屏调试和本地上位机调试两部分。触摸屏系统的调试步骤如下&#xff1a; 确认触摸屏和PLC的通讯情…

C++标准库 -- 泛型算法 (Primer C++ 第五版 · 阅读笔记)

C标准库 -- 泛型算法 (Primer C 第五版 阅读笔记&#xff09; 第10章 泛型算法------(持续更新)10.1、概述10.2、初识泛型算法10.2.1、只读算法10.2.2、写容器元素的算法10.2.3、重排容器元素的算法 10.3、定制操作10.4、再探迭代器10.5、泛型算法结构10.6、特定容器算法 第10…

【ubuntu】将硬盘挂载到指定目录并设置开机自动挂载

最近打算将数据盘开机自动挂载&#xff0c;省得每次都要手动挂载&#xff0c;总结步骤如下&#xff1a; 输入以下命令&#xff0c;将系统中所有的分区都列出来&#xff1a; sudo fdisk -l找到你要挂载的分区&#xff0c;可以通过容量大小分辨&#xff0c;当然&#xff0c;有可…

黑马Mysql从入门到高级

文章目录 1. 数据库基础1.1 基础概念1.2 SQL1.2.1 语法1.2.2 数据类型1.2.3 DDL&#xff08;definition&#xff09; 1.3 函数1.3.1 字符串函数1.3.2 日期函数1.3.3 数字函数1.3.4 流程函数 1.4 约束1.5 多表查询1.5.1 连接1.5.2 联合查询 2. 数据库进阶2.1 存储引擎2.1.1 Inno…

中国人民大学与加拿大女王大学金融硕士——学习的阶段让未来的人生更丰盈

初入职场的新人拥有同样的起跑线&#xff0c;经过时间的沉淀&#xff0c;每个人之间就会有差距。差距是怎样被拉开的呢&#xff1f;也可以说是行动导致的。毕竟想是问题&#xff0c;去做才是答案&#xff0c;有行动才会有结果。在职读研与其停留在想的阶段&#xff0c;不如去准…

我发现了PMP通关密码!这14页纸直接背!

一周就能背完的PMP考试技巧只有14页纸 共分成了4大模块 完全不用担心看不懂 01关键词篇 第1章引论 1.看到“驱动变革”--选项中找“将来状态” 2.看到“依赖关系”--选项中找“项目集管理” 3.看到“价值最大化”--选项中找“项目组合管理” 4.看到“可行性研究”--选项中…

「线性DP-步入」最长上升子序列(LIS)

题目描述 给定一个长度为 N 的数列&#xff0c;求数值严格单调递增的子序列的长度最长是多少。 输入格式 第一行包含整数 N。 第二行包含 N 个整数&#xff0c;表示完整序列。 输出格式 输出一个整数&#xff0c;表示最大长度。 数据范围 1 ≤ N ≤ 1000 1≤N≤1000 1≤…

【进阶C语言】有关动态内存管理的经典笔试题(详细图文讲解)

前言 &#x1f4d5;作者简介&#xff1a;热爱跑步的恒川&#xff0c;致力于C/C、Java、Python等多编程语言&#xff0c;热爱跑步&#xff0c;喜爱音乐的一位博主。 &#x1f4d7;本文收录于C语言进阶系列&#xff0c;本专栏主要内容为数据的存储、指针的进阶、字符串和内存函数…

.class文件在线转成.java文件方法

使用背景&#xff1a; 工作中碰到老的项目就头疼&#xff0c;有些连源码都没有&#xff0c;解决bug只能从class文件反编译成java后&#xff0c;来读懂业务逻辑。 后来在网上找了一些工具来帮助class文件转码&#xff0c;有些遇到lamda表达式就转换错误&#xff0c;有些使用起…

正则表达式的字符串取反常用正则表达式

正则表达式的字符串取反操作 文件同步时&#xff0c;想要过滤掉扩展名为.tmp或者.TMP的临时文件&#xff0c;想要使用正则表达式对字符串进行取反操作。 注意&#xff1a;[^tmp]* 这种取反的表达式&#xff0c;只能表示匹配除了t、m、p以外的所有字符&#xff0c;是单字符匹配…

gRPC-Go源码解读三 服务端处理流程分析

相较于Client端的复杂处理流程&#xff0c;Server端相对来说简单了很多&#xff0c;核心就是创建个TCP套接字并监听&#xff0c;收到客户端连接请求则起个go协程处理&#xff0c;子协程根据请求中的服务名和方法名调用对应的服务方法处理&#xff0c;处理完成之后则返回响应。整…

叫板IT部门和专业软件公司,低代码成为企业数字化的新选择

从2017年政府将“数字经济”写入工作报告&#xff0c;到今年两会将企业数字化转型列为重点议题&#xff0c;数字化的口号已喊了6年。政策对于数字化的支持越来越坚定&#xff0c;令人欣喜的是&#xff0c;越来越多具有远见卓识的企业已将数字化建设作为工作重心。 然而&#xf…

【LeetCode】剑指 Offer 60. n个骰子的点数 p294 -- Java Version

题目链接&#xff1a;https://leetcode.cn/problems/nge-tou-zi-de-dian-shu-lcof/ 1. 题目介绍&#xff08;60. n个骰子的点数&#xff09; 把n个骰子扔在地上&#xff0c;所有骰子朝上一面的点数之和为 s。输入 n&#xff0c;打印出 s 的所有可能的值出现的概率。 你需要用一…