C#winform上位机开发学习笔记3-串口助手的信息保存功能添加

1.功能描述

下次开启串口助手时会存留上次的发送框信息,以及选择的串口号(备份串口号与发送栏内容

原理:将文本信息保存在,ini文件中,下次打开软件时读取文件信息刷新对应控件文本

注意:在窗口关闭事件函数中也可进行信息备份工作,但不稳定,因此用本篇中的方法进行较为可靠

2.代码部分

2.1 创建文件与编辑写入信息存储代码

2.1.1 public partial class Form1 : Form初始化程序中加入以下代码

using System.Runtime.InteropServices;//应用空间变量声明,未声明则出现[DllImport("kernel32")]报错public partial class Form1 : Form{//信息保存功能[DllImport("kernel32")]//写文件函数private static extern long WritePrivateProfileString(string section, string key, string val, string filePath);//系统dll导入ini写函数[DllImport("kernel32")]//读文件函数private static extern int GetPrivateProfileString(string section, string key, string def, StringBuilder retVal, int size, string filePath);//系统dll导入ini读函数//创建文件,文件名为:Backup.inistring FileName = System.AppDomain.CurrentDomain.BaseDirectory + "Backup.ini";}

 #存储的文件保存在bin文件中,文件名为创建时编辑的名称“Backup.ini”

2.1.2 串口发送栏函数

备份发送栏

 //串口发送栏函数private void textBox5_TextChanged(object sender, EventArgs e){//备份发送栏WritePrivateProfileString("串口1", "发送栏", textBox5.Text, FileName);//保存文件:其中串口1与发送栏为文本内的标识,将发送栏的信息存储在filename文件中}

2.1.3 手动扫描并添加串口号函数中添加备份串口号程序

备份串口号

 //备份端口号WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);//保存文件:其中串口1与端口号为文本内的标识,将COM组合框的信息存储在filename文件中

手动扫描并添加串口号函数完整代码如下:

        //自定义函数1-手动扫描并添加可用串口private void SearchAndAddSerialToComboBox(SerialPort MyPort, ComboBox MyBox){string[] ComputerPortName = SerialPort.GetPortNames();  //获取本机串口列表//string BackupPort;//定义字符串备份端口号bool MyBackupPortFlag = false;  //备份端口号可用标志位//恢复端口号GetPrivateProfileString("串口1", "端口号", "", BackupBuf, 100, FileName);  //获取备份的端口号string MyBackupPortName = BackupBuf.ToString();//定义字符串备份端口号//添加端口号MyBox.Items.Clear();  //清空ComboBox内容for (int i = 0; i < ComputerPortName.Length; i++) //循环{if (MyBackupPortName == ComputerPortName[i])  //之前的端口号可用,置位标志位{MyBackupPortFlag = true;}try  //核心原理是依靠try和catch完成遍历{MyPort.PortName = ComputerPortName[i];MyPort.Open();                          //如果失败,后面的代码不会执行          MyBox.Items.Add(MyPort.PortName);       //打开成功,添加至下拉列表MyPort.Close();                         //关闭//检测备份端口号是否有效,端口号初始化为备份端口号if (MyBackupPortName == MyPort.PortName){comboBox1.Text = MyBackupPortName;}//如果备份端口号无效,端口号初始化为第一个可用端口号if (comboBox1.Text == " ")comboBox1.Text = MyPort.PortName;}catch{}}//端口号赋初值if (MyBackupPortFlag == true){MyBox.Text = MyBackupPortName; //添加备份的端口号}else{if (ComputerPortName.Length > 0){MyBox.Text = ComputerPortName[ComputerPortName.Length - 1]; //添加数值大的端口号}else{textBox1.AppendText("[" + DateTime.Now.ToString("HH:mm:ss") + "]" + "->");textBox1.AppendText("没有检测到串口工具!\r\n");}}//备份端口号WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);//保存文件:其中串口1与端口号为文本内的标识,将COM组合框的信息存储在filename文件中}

复选框函数中也添加此程序代码

private void comboBox1_SelectedIndexChanged(object sender, EventArgs e){//备份端口号WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);//保存文件:其中串口1与端口号为文本内的标识,将COM组合框的信息存储在filename文件中}

复选框函数万整改代码如下:

        //串口开启状态下切换串口号private void comboBox1_SelectedIndexChanged(object sender, EventArgs e){if (serialPort1.IsOpen) //串口已打开,此时关闭串口 {serialPort1.Close();                       //打开端口try{serialPort1.PortName = comboBox1.Text;    //端口号serialPort1.Open();                       //打开端口textBox1.AppendText("[" + DateTime.Now.ToString("HH:mm:ss") + "]" + "->");textBox1.AppendText("串口更换成功!\r\n");WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);}catch{serialPort1.Close();button2.BackgroundImage = Properties.Resources.Image_CloseSerial;button2.Tag = "OFF";textBox1.AppendText("[" + DateTime.Now.ToString("HH:mm:ss") + "]" + "->");textBox1.AppendText("串口更换失败!\r\n");}}//备份端口号WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);//保存文件:其中串口1与端口号为文本内的标识,将COM组合框的信息存储在filename文件中}

2.2 编写读取文件赋值文本代码

2.2.1 恢复发送栏

2.2.1.1 初始化函数中加入下述代码
    public partial class Form1 : Form{//定义字符串用于存储读出的.ini文件中的内容变量,大小为100个字节StringBuilder BackupBuf = new StringBuilder(100);}
2.2.1.2 窗体装载函数中加入下述代码
        //窗体装载函数private void Form1_Load(object sender, EventArgs e){//恢复发送栏//获取FileName中存储在BackupBuf中内存为100字节的标识为"串口1", "发送栏"的变量,恢复内容若没有则获取空字符GetPrivateProfileString("串口1", "发送栏", "", BackupBuf, 100, FileName);textBox5.Text = BackupBuf.ToString();//获取后将其转换为字符串赋值给发送栏}
2.2.3 恢复时判断存储的端口号是否有效,有效则恢复存储文件的端口号,无效则恢复为首端口号
        //自定义函数1-手动扫描并添加可用串口private void SearchAndAddSerialToComboBox(SerialPort MyPort, ComboBox MyBox){//恢复端口号GetPrivateProfileString("串口1", "端口号", "", BackupBuf, 100, FileName);  //获取备份的端口号string MyBackupPortName = BackupBuf.ToString();//定义字符串备份端口号//检测备份端口号是否有效,端口号初始化为备份端口号if (MyBackupPortName == MyPort.PortName){comboBox1.Text = MyBackupPortName;}//如果备份端口号无效,端口号初始化为第一个可用端口号if (comboBox1.Text == " ")comboBox1.Text = MyPort.PortName;   }         

手动扫描并添加串口号函数完整代码如下:

        //自定义函数1-手动扫描并添加可用串口private void SearchAndAddSerialToComboBox(SerialPort MyPort, ComboBox MyBox){string[] ComputerPortName = SerialPort.GetPortNames();  //获取本机串口列表//string BackupPort;//定义字符串备份端口号bool MyBackupPortFlag = false;  //备份端口号可用标志位//恢复端口号GetPrivateProfileString("串口1", "端口号", "", BackupBuf, 100, FileName);  //获取备份的端口号string MyBackupPortName = BackupBuf.ToString();//定义字符串备份端口号//添加端口号MyBox.Items.Clear();  //清空ComboBox内容for (int i = 0; i < ComputerPortName.Length; i++) //循环{if (MyBackupPortName == ComputerPortName[i])  //之前的端口号可用,置位标志位{MyBackupPortFlag = true;}try  //核心原理是依靠try和catch完成遍历{MyPort.PortName = ComputerPortName[i];MyPort.Open();                          //如果失败,后面的代码不会执行          MyBox.Items.Add(MyPort.PortName);       //打开成功,添加至下拉列表MyPort.Close();                         //关闭//检测备份端口号是否有效,端口号初始化为备份端口号if (MyBackupPortName == MyPort.PortName){comboBox1.Text = MyBackupPortName;}//如果备份端口号无效,端口号初始化为第一个可用端口号if (comboBox1.Text == " ")comboBox1.Text = MyPort.PortName;}catch{}}//端口号赋初值if (MyBackupPortFlag == true){MyBox.Text = MyBackupPortName; //添加备份的端口号}else{if (ComputerPortName.Length > 0){MyBox.Text = ComputerPortName[ComputerPortName.Length - 1]; //添加数值大的端口号}else{textBox1.AppendText("[" + DateTime.Now.ToString("HH:mm:ss") + "]" + "->");textBox1.AppendText("没有检测到串口工具!\r\n");}}//备份端口号WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);//保存文件:其中串口1与端口号为文本内的标识,将COM组合框的信息存储在filename文件中}

3.涉及到的相关完整代码

    public partial class Form1 : Form{//信息保存功能[DllImport("kernel32")]//写文件函数private static extern long WritePrivateProfileString(string section, string key, string val, string filePath);//系统dll导入ini写函数[DllImport("kernel32")]//读文件函数private static extern int GetPrivateProfileString(string section, string key, string def, StringBuilder retVal, int size, string filePath);//系统dll导入ini读函数//创建文件,文件名为:Backup.inistring FileName = System.AppDomain.CurrentDomain.BaseDirectory + "Backup.ini";//定义字符串用于存储读出的.ini文件中的内容变量,大小为100个字节StringBuilder BackupBuf = new StringBuilder(100);public Form1(){InitializeComponent();//串口接收支持中文显示serialPort1.Encoding = Encoding.GetEncoding("GB2312");  //串口1的解码支持GB2312汉字System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false;//检查跨线程出错解决方法}//窗体装载函数private void Form1_Load(object sender, EventArgs e){SearchAndAddSerialToComboBox(serialPort1, comboBox1);//扫描端口号,并将可用端口添加至下拉列表comboBox14.Text = "115200";                           //默认显示值this.button2.PerformClick();                         //自动打开串口//恢复发送栏//获取FileName中存储在BackupBuf中内存为100字节的标识为"串口1", "发送栏"的变量,恢复内容若没有则获取空字符GetPrivateProfileString("串口1", "发送栏", "", BackupBuf, 100, FileName);textBox5.Text = BackupBuf.ToString();//获取后将其转换为字符串赋值给发送栏}//自定义函数1-手动扫描并添加可用串口private void SearchAndAddSerialToComboBox(SerialPort MyPort, ComboBox MyBox){string[] ComputerPortName = SerialPort.GetPortNames();  //获取本机串口列表//string BackupPort;//定义字符串备份端口号bool MyBackupPortFlag = false;  //备份端口号可用标志位//恢复端口号GetPrivateProfileString("串口1", "端口号", "", BackupBuf, 100, FileName);  //获取备份的端口号string MyBackupPortName = BackupBuf.ToString();//定义字符串备份端口号//添加端口号MyBox.Items.Clear();  //清空ComboBox内容for (int i = 0; i < ComputerPortName.Length; i++) //循环{if (MyBackupPortName == ComputerPortName[i])  //之前的端口号可用,置位标志位{MyBackupPortFlag = true;}try  //核心原理是依靠try和catch完成遍历{MyPort.PortName = ComputerPortName[i];MyPort.Open();                          //如果失败,后面的代码不会执行          MyBox.Items.Add(MyPort.PortName);       //打开成功,添加至下拉列表MyPort.Close();                         //关闭//检测备份端口号是否有效,端口号初始化为备份端口号if (MyBackupPortName == MyPort.PortName){comboBox1.Text = MyBackupPortName;}//如果备份端口号无效,端口号初始化为第一个可用端口号if (comboBox1.Text == " ")comboBox1.Text = MyPort.PortName;}catch{}}//端口号赋初值if (MyBackupPortFlag == true){MyBox.Text = MyBackupPortName; //添加备份的端口号}else{if (ComputerPortName.Length > 0){MyBox.Text = ComputerPortName[ComputerPortName.Length - 1]; //添加数值大的端口号}else{textBox1.AppendText("[" + DateTime.Now.ToString("HH:mm:ss") + "]" + "->");textBox1.AppendText("没有检测到串口工具!\r\n");}}//扫描后写入备份端口号WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);//保存文件:其中串口1与端口号为文本内的标识,将COM组合框的信息存储在filename文件中}//自定义函数2-CRC校验private UInt16 Crc_Check(byte[] Data, byte DataLEN){UInt16 CRC = 0xFFFF;for (byte i = 0; i < DataLEN; i++){ CRC ^= Data[i];for(byte j = 0; j < 8; j++){if((CRC & 0x0001) == 0x0001){CRC = (UInt16)((CRC >> 1) ^ 0xA001);}else{CRC = (UInt16)(CRC >> 1);}}}CRC = (UInt16)((CRC >> 8) + (CRC<<8));return CRC;}//串口开启状态下切换串口号private void comboBox1_SelectedIndexChanged(object sender, EventArgs e){if (serialPort1.IsOpen) //串口已打开,此时关闭串口 {serialPort1.Close();                       //打开端口try{serialPort1.PortName = comboBox1.Text;    //端口号serialPort1.Open();                       //打开端口textBox1.AppendText("[" + DateTime.Now.ToString("HH:mm:ss") + "]" + "->");textBox1.AppendText("串口更换成功!\r\n");WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);}catch{serialPort1.Close();button2.BackgroundImage = Properties.Resources.Image_CloseSerial;button2.Tag = "OFF";textBox1.AppendText("[" + DateTime.Now.ToString("HH:mm:ss") + "]" + "->");textBox1.AppendText("串口更换失败!\r\n");timer1.Stop();}}//复选框选择后备份端口号WritePrivateProfileString("串口1", "端口号", comboBox1.Text, FileName);//保存文件:其中串口1与端口号为文本内的标识,将COM组合框的信息存储在filename文件中}//串口发送栏函数private void textBox5_TextChanged(object sender, EventArgs e){//备份发送栏WritePrivateProfileString("串口1", "发送栏", textBox5.Text, FileName);//保存文件:其中串口1与发送栏为文本内的标识,将发送栏的信息存储在filename文件中}
}

4.测试结果

关闭软件后重新打开发送框与com口选择框出现关闭前输入的信息,功能添加成功

此功能涉及到的代码较多,且多与其他功能函数相关,总体的逻辑为创建文件->将需要备份的信息存储在文件内->下次开启软件后自动从备份的文件中读取信息并赋值到相关的文本中,从而达到信息保存的目的

注:参考自B站硬件家园

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

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

相关文章

流量控制与熔断利器:Sentinel介绍

这是《百图解码支付系统设计与实现》专栏系列文章中的第&#xff08;19&#xff09;篇&#xff0c;也是流量控制系列的第&#xff08;6&#xff09;篇。点击上方关注&#xff0c;深入了解支付系统的方方面面。 本篇聊聊流量控制与熔断利器Sentinel&#xff0c;背后的原理&…

AWS 专题学习 P5 (Classic SA、S3)

文章目录 Classic Solutions Architecture无状态 Web 应用程序&#xff1a;WhatIsTheTime.com背景 & 目标架构演进Well-Architected 5 pillars 有状态的 Web 应用程序&#xff1a;MyClothes.com背景 & 目标架构演进总结 有状态的 Web 应用程序&#xff1a;MyWordPress.…

【51单片机】动态数码管

0、前言 参考&#xff1a; 普中51单片机开发攻略–A2.pdf 1、数码管介绍 上一章我们主要是介绍一位数码管的内部结构及控制原理。下面我们再来介 绍下多位数码管及动态显示原理的相关知识。 1.1 多位数码管简介 2、74HC245 和 74HC138 芯片介绍 2.1 74HC245 芯片简介 2.2 7…

数据结构排序二叉树(下)

哎,调了几天深度学习模型,今天来更新排序二叉树 文章目录 前言 一、排序二叉树的结构定义 二、在排序二叉树添加数据 三、定义创建排序二叉树函数 四、查找一棵二叉排序树中的结点x的所在层数 五、删除二叉排序树中T关键字x的节点 六、查找二叉排序树中的所有小于key的关…

数字化变革时代|打破瓶颈 海格里斯HEGERLS四向穿梭车系统技术的新突破

数字化转型是国内外市场环境的必然趋势, 从各大中小企业的创新驱动力来说&#xff0c;物联网、云计算、人工智能、大数据等等&#xff0c;都是创新的技术革命&#xff0c;也是当前大环境下所追寻的数字化变革。面对数字化制造时代的到来&#xff0c;河北沃克不失时机地布局了自…

【人工智能平台】ubuntu22.04.3部署cube-studio

简介&#xff1a;本次安装是在虚拟机上进行&#xff0c;需要给虚拟机至少分配16GB&#xff0c;分配8GB时系统会卡死。 一、环境&#xff1a; 主机环境&#xff1a;win11&#xff08;全程科学&#xff09;vm虚拟机 虚拟机&#xff1a;ubuntu22.04.3桌面版&#xff08;新装&…

Unity vs Godot :哪个游戏引擎更适合你?

Unity vs Godot &#xff1a;哪个游戏引擎更适合你&#xff1f; 游戏引擎的选择对开发过程和最终产品质量有着重大影响。近年来&#xff0c;Godot和Unity这两款引擎受到广泛关注。本文将从多个维度对两者进行比较&#xff0c;以期为开发者提供正确的选择建议。 Godot和Unity都有…

Spring的@Async使用防坑

很多人会直接只用Async来执行异步操作。但是这里面有一个问题&#xff0c;如果都是用spring的默认实现&#xff0c;是有坑的。 当我们开启 EnableAsync注解 就可以使用Async注解来执行异步操作了。 我们现在通过EnableAsync注解去打开spring异步世界的坑。 走JDK的动态代理&a…

力扣 | 11. 盛最多水的容器

双指针解法–对撞指针 暴力解法public int maxArea1(int[] height) {int n height.length;int ans 0;for (int i 0; i < n; i) {for (int j i 1; j < n; j) {int area Math.min(height[i], height[j]) * (j - i);ans Math.max(ans, area);}}return ans;}双指针解法…

探索curl的高级应用:HTTP请求的大师级技巧

探索curl的高级应用&#xff1a;HTTP请求的大师级技巧 引言高级用法概览1. HTTP请求与响应处理2. 身份验证与安全3. 进阶技巧4. Cookie管理与会话保持5. 脚本自动化 HTTP请求与响应处理1. 自定义请求头2. 发送数据3. 处理响应 身份验证与安全1. 基本认证2. 摘要认证3. HTTPS安全…

MyBatis框架基础到进阶

1、为什么要学习MyBatis 如果没有MyBatis框架&#xff0c;我们依靠JDBC和连接池已经能够很好的和数据库进行交互了&#xff0c;而学习MyBatis框架最核心的原因是为了减少SQL语句对代码的侵入性。 因为在过往不管是使用连接池还是JDBC Templete&#xff0c;所有的SQL语句都写在代…

Docker项目部署()

1.创建文件夹tools mkdir tools 配置阿里云 Docker Yum 源 : yum install - y yum - utils device - mapper - persistent - data lvm2 yum - config - manager -- add - repo http://mirrors.aliyun.com/docker- ce/linux/centos/docker - ce.repo 更新 yum 缓存 yum makec…