基于python+pyqt5的串口助手

基于python+pyqt5的串口助手

环境: pycharm、python3.8,pyqt5,pyserial。(需要该节的工程文件 请私信,或加VX:Crazzy_M,订阅后直接拿源码)

说明: 通过python3.8、pyqt5和pyserial自制串口助手,可以实现基础ascll码,hex数据的收发以及定时发送,还可以实现接收数据的动态波形显示。

功能展示

在工程应用中串口助手的其他功能因项目而异,因此学会基础功能的搭建,在改进的时候就方便很多。

请添加图片描述


1.环境的安装(很重要)

   Python3.8和pycharm进行常规安装即可(pycharm需要破解一下)。(如果需要安装包可以私信)

1.1.所需包的安装

所需的包

说明
pyqt5链接QT
pyqt5-tools链接QT
pyserial串口库
matplotlib绘图所需的matlib包
pyinstaller生成EXE所需的包
py2exe生成exe所需的包(关系到exe任务栏和左上角图标能否正常显示

安装方法(下面三个选一个就可以):

①默认方法(安装比较慢):

pip install  pack    (pack是指上面的包)

②推荐使用清华源:

pip install -i https://pypi.tuna.tsinghua.edu.cn/simple/ pack     (pack是指上面的包)

③也可以使用其他源

阿里云: https://mirrors.aliyun.com/pypi/simple/  pack     (pack是指上面的包)

豆瓣:http://pypi.douban.com/simple/ pack     (pack是指上面的包)

安装好包之后

在Terminal下执行pip list可以看到安装的包

(有些包是上面总结的包附带的)
在这里插入图片描述
还有下面这个包

pyqt5-tools



1.2. 链接QT

首先我们GUI是通过QT搭建的,界面如下所示:

在这里插入图片描述

我们通过两个步骤进行python和qt的关联,也就是设置下面两个外部工具进行 QT设计师的打开QT代码的更新

在这里插入图片描述

1.2.1 pycharm如何调用QT设计师

我们要通过pycharm去调用QT设计师去设计GUI界面,但是在pycharm中如何直接调用QT设计师呢?

前提条件是已经安装上面所述的pyqt5的包。

接下来就配置 QtDesigner 外部工具,用来调用QT界面:
在这里插入图片描述


file→settings
在这里插入图片描述

Settings→Tools→External Tools(扩展包)

里面的QtDesigner和pyuic5是我已经配置好的,没有配置里面是空白的。
在这里插入图片描述
③配置QtDesigner

点击“+”号就会出现以下界面

在这里插入图片描述
④开始填写

Name: QtDesigner

Program: 选择designer.exe的目录

(一般来说,在你Python安装目录下的\Lib\site-packages\pyqt5-tools\designer.exe。安装的第三方库一般都在Lib文件夹下site-packages文件夹中。装完PyQt5,和PyQt-tools后,该目录下至少会有这俩个文件夹)。

注意 新版本的python下designer.exe的目录在安装Pyqt5-tools后,designer.exe所在路径是python\Lib\site-packages\qt5_applications\Qt\bin\designer.exe(安装目录下找)

我的路径是:

D:\software\python\Anaconda3\Lib\site-packages\qt5_applications\Qt\bin\designer.exe

在这里插入图片描述
Arguments: $FileName$ (复制吧,也可以自己选择insert插入,这样保证已经生成的ui文件直接用designer直接打开)

Working directory: $FileDir$ (路径设置,同样可以直接复制,选完program后会把它当默认路径而不是工程路径,这个参数会让路径转为工程路径)

配置完界面如下:
在这里插入图片描述
配置完成

在Tools→External Tools中点击QtDesigner就可以打开QT设计师

在这里插入图片描述
在这里插入图片描述



1.2.2 pycharm中如何更新QT对应的py代码

Qt设计师中,界面设计完毕后怎么转换为py代码呢?

同样需要配置一个外部工具(Pyuic5):
在这里插入图片描述

file→settings
在这里插入图片描述

Settings→Tools→External Tools(扩展包)

里面的QtDesigner和pyuic5是我已经配置好的,没有配置里面是空白的。
在这里插入图片描述
③配置pyuic5

Name: Pyuic5(其实名字随你,你自己懂就行了)

Program: 这里选的是你Python安装目录,找到python.exe(例如我的是D:\software\python\Anaconda3\python.exe)

Arguments: -m PyQt5.uic.pyuic $FileName$ -o $FileNameWithoutExtension$.py
Working directory: $FileDir$ (这样保证转换的文件是在你的工程目录下的,否则是其他目录)

配置好的界面如下
在这里插入图片描述

配置完成

点击Tools→External Tools→pyuic5就可以将QT设计师中GUI界面转换为py文件
在这里插入图片描述



2.串口助手代码

2.1 主代码

需要工程文件,私信。

import sys
import serial
import serial.tools.list_ports


from PyQt5.QtWidgets import QMessageBox
from PyQt5.QtCore import QTimer
from ui_demo_1 import Ui_Form

from PyQt5.QtGui import QIcon

from PyQt5.QtWidgets import QFileDialog
from matplotlib.backends.backend_qt5agg import FigureCanvasQTAgg as FigureCanvas, NavigationToolbar2QT as NavigationToolbar

from matplotlib.figure import Figure

import images

import ctypes
ctypes.windll.shell32.SetCurrentProcessExplicitAppUserModelID("myappid")

from PyQt5 import QtWidgets                     #包含构建界面的UI元素组件




class Pyqt5_Serial(QtWidgets.QWidget, Ui_Form):
    def __init__(self):                         #开头都这么写
        super(Pyqt5_Serial, self).__init__()    #开头都这么写
        self.setupUi(self)                      #开头都这么写
        self.init()                             #调用下面的init函数
        self.setWindowTitle("Ucom v1.2")        #设置标题
        self.ser = serial.Serial()
        self.port_check()                       #串口检测
        self.setWindowIcon(QIcon(':/Ucom.ico'))   #设置图标

        # 接收数据和发送数据数目置零
        self.data_num_received = 0              #接收数据清零
        self.lineEdit.setText(str(self.data_num_received))
        self.data_num_sended = 0                #发送数据清零
        self.lineEdit_2.setText(str(self.data_num_sended))
        self.ReceNumForClear = 0                # 接收框的数据个数用于自动清除  放到初始化只会初始化一次

        # 设置选项卡的名称
        self.tabWidget.setTabText(0, '数据接收')
        self.tabWidget.setTabText(1, '波形显示')
        # index = self.tabWidget.currentIndex()     #可以读取选项卡的状态




        # 添加图像显示的画布
        self.static_canvas = FigureCanvas(Figure())                     # 画布、渲染器
        layout = QtWidgets.QVBoxLayout(self.groupBox)                   # 添加垂直布局类groupBox
        layout.addWidget(self.static_canvas)                            # 向布局groupBox_1中添加渲染器
        tool_bar = NavigationToolbar(self.static_canvas, self.groupBox)  # 生成画布相关联的工具栏
        layout.addWidget(tool_bar)                                      # 向布局groupBox_2中添加工具栏
        self._static_ax1 = self.static_canvas.figure.subplots(1, 1)     # 从渲染器中的画布figure中,获取子布,也就是Axes(1行1列)


    def init(self):
        # 串口检测按钮
        self.s1__box_1.clicked.connect(self.port_check)             # 单击 检测串口 按钮链接到port_check函数

        # 串口信息显示
        self.s1__box_2.currentTextChanged.connect(self.port_imf)    # 当前文本改变(串口选择) 链接到串口信息函数

        # 打开串口按钮
        self.open_button.clicked.connect(self.port_open)            # 单击 打开串口 按钮链接到打开串口函数

        # 关闭串口按钮
        self.close_button.clicked.connect(self.port_close)          # 单击 关闭串口 按钮链接到关闭串口函数

        # 发送数据按钮
        self.s3__send_button.clicked.connect(self.data_send)        # 单击 发送 按钮链接到数据发送函数


        # 定时发送数据
        self.timer_send = QTimer()
        self.timer_send.timeout.connect(self.data_send)
        self.timer_send_cb.stateChanged.connect(self.data_send_timer)

        # 定时器接收数据
        self.tit = 0
        self.timer = QTimer(self)
        self.timer.timeout.connect(self.data_receive)              # 串口中断是2ms进一次

        # 清除发送窗口
        self.s3__clear_button.clicked.connect(self.send_data_clear)     #单击 清除 按钮链接到清除显示函数(清除发送区)

        # 清除接收窗口
        self.s2__clear_button.clicked.connect(self.receive_data_clear)  #单击 清除 按钮链接到清除显示函数(清除接收区)

        # 打开文件
        self.s3__openfile_button.clicked.connect(self.openFile)         # 单击 打开文件

        self.timer1 = QTimer()                                          # 调用QT的定时器,用于波形显示


    # ------------------按下检测串口按钮
    def port_check(self):
        # 检测所有存在的串口,将信息存储在字典中
        self.Com_Dict = {}                                              #创建一个字典,字典是可变的容器
        port_list = list(serial.tools.list_ports.comports())            #list是序列,一串数据,可以追加数据
        self.s1__box_2.clear()                                          #s1__box_2为串口选择列表
        for port in port_list:
            self.Com_Dict["%s" % port[0]] = "%s" % port[1]
            self.s1__box_2.addItem(port[0])                             #将检测到的串口放置到s1__box_2串口选择列表
        if len(self.Com_Dict) == 0:
            self.state_label.setText(" 无串口")

    # ------------------串口选择下拉框选择com口
    def port_imf(self):
        # 显示选定的串口的详细信息
        imf_s = self.s1__box_2.currentText()                            #当前显示的com口
        if imf_s != "":
            self.state_label.setText(self.Com_Dict[self.s1__box_2.currentText()])#state_label显示窗口显当前串口

    # -------------------打开串口
    def port_open(self):
        self.ser.port = self.s1__box_2.currentText()                    #串口选择框
        self.ser.baudrate = int(self.s1__box_3.currentText())           #波特率输入框
        self.ser.bytesize = int(self.s1__box_4.currentText())           #数据位输入框
        self.ser.stopbits = int(self.s1__box_6.currentText())           #停止位输入框
        self.ser.parity = self.s1__box_5.currentText()                  #校验位输入框
        try:
            self.ser.open()
        except:
	        QMessageBox.critical(self, "Port Error", "此串口不能被打开!")
	        return None


        self.timer.start(2)                                             #打开串口接收定时器,周期为2ms

        if self.ser.isOpen():                                           #打开串口按下,禁用打开按钮,启用关闭按钮
            self.open_button.setEnabled(False)                          #禁用打开按钮
            self.close_button.setEnabled(True)                          #启用关闭按钮
            self.formGroupBox1.setTitle("串口状态(已开启)")              #GroupBox1控件在串口打开的时候显示(已开启)

    # --------------------关闭串口
    def port_close(self):
        self.timer.stop()                                               # 停止计时器
        self.timer_send.stop()                                          # 停止定时发送
        self.timer1.stop()                                              # 停止图形显示计时器
        try:
            self.ser.close()
        except:
	        pass
        self.open_button.setEnabled(True)                               #启用打开按钮
        self.close_button.setEnabled(False)                             #禁用停止按钮
        self.lineEdit_3.setEnabled(True)                                #启用定时发送时间框
        # 接收数据和发送数据数目置零
        self.data_num_received = 0
        self.lineEdit.setText(str(self.data_num_received))              #接收数目置零
        self.data_num_sended = 0
        self.lineEdit_2.setText(str(self.data_num_sended))              #发送数目清零
        self.formGroupBox1.setTitle("串口状态(已关闭)")                  #GroupBox1控件在串口打开的时候显示(已关闭)

    # ----------------------发送数据
    def data_send(self):
        if self.ser.isOpen():                                           #判断串口是否打开 ↓
            input_s = self.s3__send_text.toPlainText()                  #(发送区)获取文本内容
            if input_s != "":                                           # 非空字符串 ↓
                # hex发送
                if self.hex_send.isChecked():                           #(勾选Hex发送) ↓(复选框选中返回Ture)
                    input_s = input_s.strip()                           #移除字符串头尾指定的字符(默认为空格或换行符)
                    send_list = []                                      #创建一个列表
                    while input_s != '':                                #发送框不是空的说明有数据,等发送完毕跳出循环
                        try:                                            #捕获异常
                            num = int(input_s[0:2], 16)                 #十六进制(取前两个字节数据)
                        except ValueError:
                            QMessageBox.critical(self, 'wrong data', '请输入十六进制数据,以空格分开!')
                            return None
                        input_s = input_s[2:].strip()                   #移除字符串头尾指定的字符(默认为空格或换行符)
                        send_list.append(num)                           #列表末尾添加新的对象
                    input_s = bytes(send_list)                          #返回字节对象
                # ascii发送
                else:
                    input_s = (input_s + '\r\n').encode('utf-8')        #

                num = self.ser.write(input_s)                           #串口写,返回的写入的数据数
                self.data_num_sended += num                             #发送数据统计
                self.lineEdit_2.setText(str(self.data_num_sended))      #已发送数据显示框
        else:
	        pass #空语句 保证结构的完整性

    # ----------------------接收数据
    def data_receive(self):
        try:
            num = self.ser.inWaiting()                                  # 获取接受缓存中的字符数
        except:
            self.port_close()
            return None

        if num > 0:                                                     # 如果收到数据 数据以十六进制的形式放到data里面     ,接收一次数据进一次(也就是发送端点一次发送进一次)
            data = self.ser.read(num)                                   # 从串口读取指定字节大小的数据
            num = len(data)                                             # 等到收到数据的长度
            self.recevive_num = num
            # hex显示
            if self.hex_receive.checkState():                           # 如果勾选HEX接收
                out_s = ''                                              # 如果是空的,显示空格
                self.display = []                                       # 创建一个列表,用于暂存波形波形显示
                self.display2 = []                                      # 创建一个列表,用于图形绘制
                for i in range(0, len(data)):
                    out_s = out_s + '{:02X}'.format(data[i]) + ' '      # 字符串操作显示到接收框(格式为2位十六进制)
                    self.display.append(data[i])
                self.s2__receive_text.insertPlainText(out_s)            # 接收的数据显示至接收区

                # 调用定时器实现图形绘制
                self.timer1.start(100)  # 单位为ms,10ms刷新一次

                self.i = 0  # 新建变量
                self.t = []  # 新建一个列表
                self.s = []  # 新建一个列表

                self.timer1.timeout.connect(self.refresh_plot)  # 定时器计数达到后调用函数refresh_plot


            # 如果不勾选HEX接收
            else:
                # 串口接收到的字符串为b'123',要转化成unicode字符串才能输出到窗口中去
                self.s2__receive_text.insertPlainText(data.decode('iso-8859-1'))    #接收区

            # 统计接收字符的数量
            self.data_num_received += num                               # 更新接收到所有数据的数目
            self.ReceNumForClear += num                                 # 更新接收框里面的数据的数据
            self.lineEdit.setText(str(self.data_num_received))          # 显示接收数据个数
            if self.SetAutoClear.checkState():                          # 如果勾选自动清除
                ValueClearNumSet = int(self.ClearNumSet.text())         # 获取设定的自动清除的num值
                if self.ReceNumForClear >= ValueClearNumSet:            # 如果收到的数据大于等于ValueClearNumSet 有缺陷必须是等于才会清掉
                    self.s2__receive_text.setText("")                   # 清楚接收框 不清计数
                    self.ReceNumForClear = 0                            # ReceNumForClear清零用于下一次统计接收框接收数据个数
                    self.ReceNumForClear = self.ReceNumForClear         # 调试用

            # 获取到text光标
            textCursor = self.s2__receive_text.textCursor()             #获取接收区光标
            # 滚动到底部
            textCursor.movePosition(textCursor.End)                     #
            # 设置光标到text中去
            self.s2__receive_text.setTextCursor(textCursor)
        else:
            pass


    # -------------------------接收定时器达到时被调用
    def refresh_plot(self):                                             # 定时器数值达到时被调用
                                                             #
        if(self.i == self.recevive_num):                                # 画完图后关闭定时器
            self.timer1.stop()

        else:
            self.t.append(self.i)                                           # 列表后面添加i,生成横坐标
            self.display2.append(self.display[self.i])                      # 生成纵坐标
            self._static_ax1.cla()                                          # 清空子图
            self._static_ax1.plot(self.t, self.display2)                    # 绘制新的图
            self.static_canvas.draw()                                       # 更新字画布的渲染器
            self.i += 1

    # -------------------------定时发送数据
    def data_send_timer(self):
        if self.timer_send_cb.isChecked():                              # 勾选定时发送
            self.timer_send.start(int(self.lineEdit_3.text()))          # 勾选定时发送  lineEdit_3为定时时间(设置定时周期)
            self.lineEdit_3.setEnabled(False)                           # 禁用时间输入框
        else:
            self.timer_send.stop()                                      # 不勾选定时发送时,将发送定时器关闭
            self.lineEdit_3.setEnabled(True)                            # 不勾选定时发送时,使能时间输入框

    # ------------------------清除发送
    def send_data_clear(self):
        self.s3__send_text.setText("")                                  # 发送区
        self.data_num_sended = 0                                        # 清除的时候发送计数归零
        self.lineEdit_2.setText(str(self.data_num_sended))              # 已发送计数清零
    #  ------------------------清除接收
    def receive_data_clear(self):
        self.data_num_received = 0                                      # 清除的时候接收计数归零
        self.lineEdit.setText(str(self.data_num_received))              # 定时发送时间清除
        self.s2__receive_text.setText("")                               # 接收区清零



    # ------------------------打开文件
    def openFile(self):
        fname = QFileDialog.getOpenFileName(self, '打开文件', './')      # 打开文件
        if fname[0]:                                                    # fname[0]就是要打开的文件
            with open(fname[0], 'r', encoding='gb18030', errors='ignore') as f:
                self.s3__send_text.setText(f.read())                    # 将读取的文件放到发送框里面


if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    myshow = Pyqt5_Serial()     #调用class
    myshow.show()
    sys.exit(app.exec_())




2.2 链接QT的py代码

链接QT的代码(该代码是QT的UI生成的代码,不是敲出来的哈):

# -*- coding: utf-8 -*-

# Form implementation generated from reading ui file 'ui_demo_1.ui'
#
# Created by: PyQt5 UI code generator 5.9.2
#
# WARNING! All changes made in this file will be lost!

from PyQt5 import QtCore, QtGui, QtWidgets

class Ui_Form(object):
    def setupUi(self, Form):
        Form.setObjectName("Form")
        Form.setEnabled(True)
        Form.resize(727, 535)
        self.formGroupBox = QtWidgets.QGroupBox(Form)
        self.formGroupBox.setGeometry(QtCore.QRect(20, 20, 167, 301))
        self.formGroupBox.setObjectName("formGroupBox")
        self.formLayout = QtWidgets.QFormLayout(self.formGroupBox)
        self.formLayout.setContentsMargins(10, 10, 10, 10)
        self.formLayout.setSpacing(10)
        self.formLayout.setObjectName("formLayout")
        self.s1__lb_1 = QtWidgets.QLabel(self.formGroupBox)
        self.s1__lb_1.setObjectName("s1__lb_1")
        self.formLayout.setWidget(0, QtWidgets.QFormLayout.LabelRole, self.s1__lb_1)
        self.s1__box_1 = QtWidgets.QPushButton(self.formGroupBox)
        self.s1__box_1.setAutoRepeatInterval(100)
        self.s1__box_1.setDefault(True)
        self.s1__box_1.setObjectName("s1__box_1")
        self.formLayout.setWidget(0, QtWidgets.QFormLayout.FieldRole, self.s1__box_1)
        self.s1__lb_2 = QtWidgets.QLabel(self.formGroupBox)
        self.s1__lb_2.setObjectName("s1__lb_2")
        self.formLayout.setWidget(1, QtWidgets.QFormLayout.LabelRole, self.s1__lb_2)
        self.s1__box_2 = QtWidgets.QComboBox(self.formGroupBox)
        self.s1__box_2.setObjectName("s1__box_2")
        self.formLayout.setWidget(1, QtWidgets.QFormLayout.FieldRole, self.s1__box_2)
        self.s1__lb_3 = QtWidgets.QLabel(self.formGroupBox)
        self.s1__lb_3.setObjectName("s1__lb_3")
        self.formLayout.setWidget(3, QtWidgets.QFormLayout.LabelRole, self.s1__lb_3)
        self.s1__box_3 = QtWidgets.QComboBox(self.formGroupBox)
        self.s1__box_3.setObjectName("s1__box_3")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.s1__box_3.addItem("")
        self.formLayout.setWidget(3, QtWidgets.QFormLayout.FieldRole, self.s1__box_3)
        self.s1__lb_4 = QtWidgets.QLabel(self.formGroupBox)
        self.s1__lb_4.setObjectName("s1__lb_4")
        self.formLayout.setWidget(4, QtWidgets.QFormLayout.LabelRole, self.s1__lb_4)
        self.s1__box_4 = QtWidgets.QComboBox(self.formGroupBox)
        self.s1__box_4.setObjectName("s1__box_4")
        self.s1__box_4.addItem("")
        self.s1__box_4.addItem("")
        self.s1__box_4.addItem("")
        self.s1__box_4.addItem("")
        self.formLayout.setWidget(4, QtWidgets.QFormLayout.FieldRole, self.s1__box_4)
        self.s1__lb_5 = QtWidgets.QLabel(self.formGroupBox)
        self.s1__lb_5.setObjectName("s1__lb_5")
        self.formLayout.setWidget(5, QtWidgets.QFormLayout.LabelRole, self.s1__lb_5)
        self.s1__box_5 = QtWidgets.QComboBox(self.formGroupBox)
        self.s1__box_5.setObjectName("s1__box_5")
        self.s1__box_5.addItem("")
        self.formLayout.setWidget(5, QtWidgets.QFormLayout.FieldRole, self.s1__box_5)
        self.open_button = QtWidgets.QPushButton(self.formGroupBox)
        self.open_button.setObjectName("open_button")
        self.formLayout.setWidget(7, QtWidgets.QFormLayout.SpanningRole, self.open_button)
        self.close_button = QtWidgets.QPushButton(self.formGroupBox)
        self.close_button.setObjectName("close_button")
        self.formLayout.setWidget(8, QtWidgets.QFormLayout.SpanningRole, self.close_button)
        self.s1__lb_6 = QtWidgets.QLabel(self.formGroupBox)
        self.s1__lb_6.setObjectName("s1__lb_6")
        self.formLayout.setWidget(6, QtWidgets.QFormLayout.LabelRole, self.s1__lb_6)
        self.s1__box_6 = QtWidgets.QComboBox(self.formGroupBox)
        self.s1__box_6.setObjectName("s1__box_6")
        self.s1__box_6.addItem("")
        self.formLayout.setWidget(6, QtWidgets.QFormLayout.FieldRole, self.s1__box_6)
        self.state_label = QtWidgets.QLabel(self.formGroupBox)
        self.state_label.setText("")
        self.state_label.setTextFormat(QtCore.Qt.AutoText)
        self.state_label.setScaledContents(True)
        self.state_label.setAlignment(QtCore.Qt.AlignRight|QtCore.Qt.AlignTrailing|QtCore.Qt.AlignVCenter)
        self.state_label.setObjectName("state_label")
        self.formLayout.setWidget(2, QtWidgets.QFormLayout.SpanningRole, self.state_label)
        self.verticalGroupBox = QtWidgets.QGroupBox(Form)
        self.verticalGroupBox.setGeometry(QtCore.QRect(210, 20, 401, 321))
        self.verticalGroupBox.setObjectName("verticalGroupBox")
        self.verticalLayout = QtWidgets.QVBoxLayout(self.verticalGroupBox)
        self.verticalLayout.setContentsMargins(10, 10, 10, 10)
        self.verticalLayout.setObjectName("verticalLayout")
        self.tabWidget = QtWidgets.QTabWidget(self.verticalGroupBox)
        self.tabWidget.setObjectName("tabWidget")
        self.tab = QtWidgets.QWidget()
        self.tab.setObjectName("tab")
        self.s2__receive_text = QtWidgets.QTextBrowser(self.tab)
        self.s2__receive_text.setGeometry(QtCore.QRect(0, 0, 371, 261))
        self.s2__receive_text.setObjectName("s2__receive_text")
        self.tabWidget.addTab(self.tab, "")
        self.tab2 = QtWidgets.QWidget()
        self.tab2.setObjectName("tab2")
        self.groupBox = QtWidgets.QGroupBox(self.tab2)
        self.groupBox.setGeometry(QtCore.QRect(0, 0, 371, 261))
        self.groupBox.setTitle("")
        self.groupBox.setObjectName("groupBox")
        self.tabWidget.addTab(self.tab2, "")
        self.verticalLayout.addWidget(self.tabWidget)
        self.verticalGroupBox_2 = QtWidgets.QGroupBox(Form)
        self.verticalGroupBox_2.setGeometry(QtCore.QRect(210, 350, 401, 141))
        self.verticalGroupBox_2.setObjectName("verticalGroupBox_2")
        self.verticalLayout_2 = QtWidgets.QVBoxLayout(self.verticalGroupBox_2)
        self.verticalLayout_2.setContentsMargins(10, 10, 10, 10)
        self.verticalLayout_2.setObjectName("verticalLayout_2")
        self.s3__send_text = QtWidgets.QTextEdit(self.verticalGroupBox_2)
        self.s3__send_text.setObjectName("s3__send_text")
        self.verticalLayout_2.addWidget(self.s3__send_text)
        self.s3__send_button = QtWidgets.QPushButton(Form)
        self.s3__send_button.setGeometry(QtCore.QRect(620, 450, 61, 31))
        self.s3__send_button.setObjectName("s3__send_button")
        self.s3__clear_button = QtWidgets.QPushButton(Form)
        self.s3__clear_button.setGeometry(QtCore.QRect(620, 370, 61, 31))
        self.s3__clear_button.setObjectName("s3__clear_button")
        self.formGroupBox1 = QtWidgets.QGroupBox(Form)
        self.formGroupBox1.setGeometry(QtCore.QRect(20, 390, 171, 101))
        self.formGroupBox1.setObjectName("formGroupBox1")
        self.formLayout_2 = QtWidgets.QFormLayout(self.formGroupBox1)
        self.formLayout_2.setContentsMargins(10, 10, 10, 10)
        self.formLayout_2.setSpacing(10)
        self.formLayout_2.setObjectName("formLayout_2")
        self.label = QtWidgets.QLabel(self.formGroupBox1)
        self.label.setObjectName("label")
        self.formLayout_2.setWidget(0, QtWidgets.QFormLayout.LabelRole, self.label)
        self.label_2 = QtWidgets.QLabel(self.formGroupBox1)
        self.label_2.setObjectName("label_2")
        self.formLayout_2.setWidget(1, QtWidgets.QFormLayout.LabelRole, self.label_2)
        self.lineEdit = QtWidgets.QLineEdit(self.formGroupBox1)
        self.lineEdit.setObjectName("lineEdit")
        self.formLayout_2.setWidget(0, QtWidgets.QFormLayout.FieldRole, self.lineEdit)
        self.lineEdit_2 = QtWidgets.QLineEdit(self.formGroupBox1)
        self.lineEdit_2.setObjectName("lineEdit_2")
        self.formLayout_2.setWidget(1, QtWidgets.QFormLayout.FieldRole, self.lineEdit_2)
        self.hex_send = QtWidgets.QCheckBox(Form)
        self.hex_send.setGeometry(QtCore.QRect(620, 350, 71, 16))
        self.hex_send.setChecked(False)
        self.hex_send.setObjectName("hex_send")
        self.hex_receive = QtWidgets.QCheckBox(Form)
        self.hex_receive.setEnabled(True)
        self.hex_receive.setGeometry(QtCore.QRect(620, 140, 71, 16))
        self.hex_receive.setTabletTracking(False)
        self.hex_receive.setAcceptDrops(False)
        self.hex_receive.setAutoFillBackground(False)
        self.hex_receive.setChecked(False)
        self.hex_receive.setAutoRepeat(False)
        self.hex_receive.setAutoExclusive(False)
        self.hex_receive.setTristate(False)
        self.hex_receive.setObjectName("hex_receive")
        self.s2__clear_button = QtWidgets.QPushButton(Form)
        self.s2__clear_button.setGeometry(QtCore.QRect(620, 160, 61, 31))
        self.s2__clear_button.setObjectName("s2__clear_button")
        self.timer_send_cb = QtWidgets.QCheckBox(Form)
        self.timer_send_cb.setGeometry(QtCore.QRect(260, 500, 71, 16))
        self.timer_send_cb.setObjectName("timer_send_cb")
        self.lineEdit_3 = QtWidgets.QLineEdit(Form)
        self.lineEdit_3.setGeometry(QtCore.QRect(350, 500, 61, 20))
        self.lineEdit_3.setAlignment(QtCore.Qt.AlignRight|QtCore.Qt.AlignTrailing|QtCore.Qt.AlignVCenter)
        self.lineEdit_3.setObjectName("lineEdit_3")
        self.dw = QtWidgets.QLabel(Form)
        self.dw.setGeometry(QtCore.QRect(420, 500, 54, 20))
        self.dw.setObjectName("dw")
        self.label_62 = QtWidgets.QLabel(Form)
        self.label_62.setGeometry(QtCore.QRect(630, 0, 91, 20))
        self.label_62.setObjectName("label_62")
        self.label_63 = QtWidgets.QLabel(Form)
        self.label_63.setGeometry(QtCore.QRect(630, 20, 41, 16))
        self.label_63.setObjectName("label_63")
        self.SetAutoClear = QtWidgets.QCheckBox(Form)
        self.SetAutoClear.setEnabled(True)
        self.SetAutoClear.setGeometry(QtCore.QRect(620, 200, 71, 16))
        self.SetAutoClear.setTabletTracking(False)
        self.SetAutoClear.setAcceptDrops(False)
        self.SetAutoClear.setAutoFillBackground(False)
        self.SetAutoClear.setChecked(True)
        self.SetAutoClear.setAutoRepeat(False)
        self.SetAutoClear.setAutoExclusive(False)
        self.SetAutoClear.setTristate(False)
        self.SetAutoClear.setObjectName("SetAutoClear")
        self.ClearNumSet = QtWidgets.QLineEdit(Form)
        self.ClearNumSet.setGeometry(QtCore.QRect(650, 220, 41, 20))
        self.ClearNumSet.setAlignment(QtCore.Qt.AlignRight|QtCore.Qt.AlignTrailing|QtCore.Qt.AlignVCenter)
        self.ClearNumSet.setObjectName("ClearNumSet")
        self.dw_2 = QtWidgets.QLabel(Form)
        self.dw_2.setGeometry(QtCore.QRect(620, 220, 31, 20))
        self.dw_2.setObjectName("dw_2")
        self.s3__openfile_button = QtWidgets.QPushButton(Form)
        self.s3__openfile_button.setGeometry(QtCore.QRect(620, 410, 61, 31))
        self.s3__openfile_button.setObjectName("s3__openfile_button")
        self.verticalGroupBox.raise_()
        self.verticalGroupBox_2.raise_()
        self.formGroupBox.raise_()
        self.s3__send_button.raise_()
        self.s3__clear_button.raise_()
        self.formGroupBox.raise_()
        self.hex_send.raise_()
        self.hex_receive.raise_()
        self.s2__clear_button.raise_()
        self.timer_send_cb.raise_()
        self.lineEdit_3.raise_()
        self.dw.raise_()
        self.label_62.raise_()
        self.label_63.raise_()
        self.SetAutoClear.raise_()
        self.ClearNumSet.raise_()
        self.dw_2.raise_()
        self.s3__openfile_button.raise_()

        self.retranslateUi(Form)
        self.tabWidget.setCurrentIndex(0)
        QtCore.QMetaObject.connectSlotsByName(Form)

    def retranslateUi(self, Form):
        _translate = QtCore.QCoreApplication.translate
        Form.setWindowTitle(_translate("Form", "Form"))
        self.formGroupBox.setTitle(_translate("Form", "串口设置"))
        self.s1__lb_1.setText(_translate("Form", "串口检测:"))
        self.s1__box_1.setText(_translate("Form", "检测串口"))
        self.s1__lb_2.setText(_translate("Form", "串口选择:"))
        self.s1__lb_3.setText(_translate("Form", "波特率:"))
        self.s1__box_3.setItemText(0, _translate("Form", "115200"))
        self.s1__box_3.setItemText(1, _translate("Form", "2400"))
        self.s1__box_3.setItemText(2, _translate("Form", "4800"))
        self.s1__box_3.setItemText(3, _translate("Form", "9600"))
        self.s1__box_3.setItemText(4, _translate("Form", "14400"))
        self.s1__box_3.setItemText(5, _translate("Form", "19200"))
        self.s1__box_3.setItemText(6, _translate("Form", "38400"))
        self.s1__box_3.setItemText(7, _translate("Form", "57600"))
        self.s1__box_3.setItemText(8, _translate("Form", "76800"))
        self.s1__box_3.setItemText(9, _translate("Form", "12800"))
        self.s1__box_3.setItemText(10, _translate("Form", "230400"))
        self.s1__box_3.setItemText(11, _translate("Form", "460800"))
        self.s1__lb_4.setText(_translate("Form", "数据位:"))
        self.s1__box_4.setItemText(0, _translate("Form", "8"))
        self.s1__box_4.setItemText(1, _translate("Form", "7"))
        self.s1__box_4.setItemText(2, _translate("Form", "6"))
        self.s1__box_4.setItemText(3, _translate("Form", "5"))
        self.s1__lb_5.setText(_translate("Form", "校验位:"))
        self.s1__box_5.setItemText(0, _translate("Form", "N"))
        self.open_button.setText(_translate("Form", "打开串口"))
        self.close_button.setText(_translate("Form", "关闭串口"))
        self.s1__lb_6.setText(_translate("Form", "停止位:"))
        self.s1__box_6.setItemText(0, _translate("Form", "1"))
        self.verticalGroupBox.setTitle(_translate("Form", "接收区"))
        self.tabWidget.setTabText(self.tabWidget.indexOf(self.tab), _translate("Form", "Tab 1"))
        self.tabWidget.setTabText(self.tabWidget.indexOf(self.tab2), _translate("Form", "Tab 2"))
        self.verticalGroupBox_2.setTitle(_translate("Form", "发送区"))
        self.s3__send_text.setHtml(_translate("Form", "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0//EN\" 
        self.s3__send_button.setText(_translate("Form", "发送"))
        self.s3__clear_button.setText(_translate("Form", "清除发送"))
        self.formGroupBox1.setTitle(_translate("Form", "串口状态"))
        self.label.setText(_translate("Form", "已接收:"))
        self.label_2.setText(_translate("Form", "已发送:"))
        self.hex_send.setText(_translate("Form", "Hex发送"))
        self.hex_receive.setText(_translate("Form", "Hex接收"))
        self.s2__clear_button.setText(_translate("Form", "清除接收"))
        self.timer_send_cb.setText(_translate("Form", "定时发送"))
        self.lineEdit_3.setText(_translate("Form", "20"))
        self.dw.setText(_translate("Form", "ms/次"))
        self.label_62.setText(_translate("Form", "Author:Crazzy_M"))
        self.label_63.setText(_translate("Form", "V1.2"))
        self.SetAutoClear.setText(_translate("Form", "自动清除"))
        self.ClearNumSet.setText(_translate("Form", "10000"))
        self.dw_2.setText(_translate("Form", "超过"))
        self.s3__openfile_button.setText(_translate("Form", "打开文件"))



3.虚拟环境下生成exe(含图标)

3.1任务栏和exe左上角的图标添加

生成exe之前我们先给exe加个图标
在这里插入图片描述
①图标对应的图片自己选择哈,格式是ico,随便找个网站就能转换。
在这里插入图片描述


新建.qrc文件
名字自己命名
在这里插入图片描述

我命名为images.qrc
在这里插入图片描述
③images.qrc中添加代码

<RCC>
    <qresource prefix="/">
        <file>Ucom.ico</file>
     </qresource>
</RCC>

在这里插入图片描述

④生成images.py文件

执行

pyrcc5 -o images.py images.qrc

首先保证Terminal中的路径是在刚才新建的images.qrc文件下

不然会提示错误
在这里插入图片描述
通过cd指令将目录转至images.qrc文件下再执行

pyrcc5 -o images.py images.qrc

在这里插入图片描述

执行命令之前是没有images.py
在这里插入图片描述
执行命令之后
在这里插入图片描述

images.py文件中是什么呢,其实就是将图片转换为了数据
在这里插入图片描述

⑤将转换后的images.py引入主程序

import images

⑥在程序中调用

self.setWindowIcon(QIcon(':/Ucom.ico'))   #设置图标

在这里插入图片描述

至此图标设置完成(注意要安装py2exe的包,不然有可能打包出来的图标会不显示)。

3.2 虚拟环境下生成EXE

如果你在运行环境下没装过其他的包,可以直接执行

pyinstaller -F --icon=Ucom.ico pyserial_demo.py -w

但是如果你安装的多余包过多,或者是以前项目安装了很多包就会导致打包的exe文件异常大,动辄300M+。

所以我们需要创建虚拟环境进行exe的打包,也就是为exe创建一个纯净的包环境。

①安装pipenv

pip install --user pipenv

②创建虚拟环境

pipenv shell

进入虚拟环境,操作环境就会变化
在这里插入图片描述

③查看虚拟环境下有哪些包

pip list

下面是已经安装过包的list
在这里插入图片描述

④安装所需要的包
所需的包

说明
pyqt5链接QT
pyqt5-tools链接QT
pyserial串口库
matplotlib绘图所需的matlib包
pyinstaller生成EXE所需的包
py2exe生成exe所需的包(关系到exe任务栏和左上角图标能否正常显示

我是用的清华源比较快一点。

 pip install -i https://pypi.tuna.tsinghua.edu.cn/simple/ pyqt5
 pip install -i https://pypi.tuna.tsinghua.edu.cn/simple/ pyqt5-tools
 pip install -i https://pypi.tuna.tsinghua.edu.cn/simple/ pyserial
 pip install -i https://pypi.tuna.tsinghua.edu.cn/simple/ pyinstaller 
 pip install -i https://pypi.tuna.tsinghua.edu.cn/simple/ py2exe  
 ...

⑤执行打包代码

虚拟环境下要安装pyinstaller 包,不然还是利用python默认的环境打包,打包出来的文件还是很大。

pyinstaller -F --icon=Ucom.ico pyserial_demo.py -w

⑥完成

在dist文件中可以找到生成的exe文件
在这里插入图片描述

打开:

在这里插入图片描述

3.3 功能演示

功能展示

★★★如有错误,欢迎指导!!!

  • 33
    点赞
  • 129
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 29
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 29
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

数字硬鉴

你的鼓励就是我创作的最大动力。

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值