打印

使用python+qt做一个串口波形调试工具

[复制链接]
69|0
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
本帖最后由 穿西装的强子 于 2025-1-17 11:39 编辑

#申请原创# #技术资源# @21小跑堂
我们经常会用到串口打印一些数据,有时候不直观,通过曲线观察会更加清晰明了,网上有很多工具,但是总感觉不适合自己,那就让我们自己做一个适合自己的串口波形调试工具;
为什么用到Python:
Python 是一种高级编程语言,因其简洁的语法和代码可读性而广受欢迎。
  • 易学易用:

    • Python 有相对简单的语法,这使得它成为初学者学习编程的理想选择。
    • 它减少了编写程序所需的代码量,因此开发人员可以更专注于解决问题。
  • 跨平台支持:

    • Python 可以在多种操作系统上运行,如 Windows、macOS、Linux 和 Unix。
    • 编写的 Python 程序无需修改即可在不同平台上运行(大多数情况下)。
  • 丰富的标准库和第三方模块:

    • Python 拥有一个庞大的标准库,提供了许多功能,比如文件 I/O、系统调用、数**算等。
    • 大量的第三方库和框架让开发者能够快速实现各种复杂功能,例如网络爬虫、数据可视化、机器学习等。
  • 面向对象和支持多种编程范式:

    • Python 支持面向对象编程,同时也兼容过程化编程和函数式编程风格。
    • 这种灵活性允许开发者根据项目需求选择最合适的编程方式。
  • 社区支持和资源丰富:

    • Python 拥有一个活跃且友好的社区,为新手和专业人士提供帮助。
    • 社区成员不断贡献新的库、工具和教程,有助于持续学习和发展。
  • 适用于多种应用领域:

    • Python 广泛应用于 Web 开发、自动化脚本、数据分析、人工智能、机器学习等领域。
    • 它也是科学计算、教育和其他专业领域的首选语言之一。
  • 动态类型系统:

    • Python 是一种解释型语言,具有动态类型系统,这意味着变量不需要显式声明类型,简化了编码过程。
    • 不过这也可能带来性能上的劣势,因为相比静态类型语言,它的执行速度较慢。
  • 良好的集成能力:

    • Python 可以轻松与其他语言编写的代码集成,如 C/C++、Java 等。
    • 这对于需要高性能部分或已有大型代码库的项目来说非常有用。
  • 开源和免费:

    • Python 是开源软件,意味着它是免费使用的,并且用户可以根据需要查看和修改源代码。
python+qt能够跨平台和丰富的库资源,让我们开发效率事半功倍;


设计思路


1.界面内容
串口选择,波特率选择,停止位、奇偶校验等串口功能;
串口打开按键
多曲线选择
实时曲线界面
2.设计流程

3.开始编程
界面设计,设计2个combobox,一个作为串口选择,已添加打开软件自动搜索添加到combobox内,一个作为波特率选择;设计1个按键,打开/关闭串口;设计4个checkbox,选择4个曲线的开关,默认为打开
self.setGeometry(400, 200, 800, 660)  # 位置和大小
        self.setMinimumSize(QtCore.QSize(self.app_w, self.app_h))
        self.setMaximumSize(QtCore.QSize(self.app_w, self.app_h))
        ports = serial.tools.list_ports.comports()

        #### Create Gui Elements ###########
        self.mainbox = QtWidgets.QWidget()
        self.setCentralWidget(self.mainbox)
        self.mainbox.setLayout(QtWidgets.QVBoxLayout())

        self.combex = QtWidgets.QComboBox(self.mainbox)
        for port, desc, hwid in sorted(ports):
            self.combex.addItem(port)
        self.ser =serial.Serial()
        self.ser.close()
        self.combex.setGeometry(10, 0, 100, 20)
        
        self.baud = QtWidgets.QComboBox(self.mainbox)
        self.baud.setGeometry(120, 0, 100, 20)
        baud = [9600,19200,38400,57600,115200,230400,460800,921600]
        for i in baud:
            self.baud.addItem(str(i))
        self.baud.setCurrentIndex(4)
        
        self.open = QtWidgets.QPushButton("打开串口",self.mainbox)
        self.open.clicked.connect(self.open_serial)
        self.open.setGeometry(230, 0, 100, 20)
        
        self.check1 = QtWidgets.QCheckBox("曲线1",self.mainbox)
        self.check1.setGeometry(340, 0, 50, 20)
        self.check1.setStyleSheet("background-color: yellow;")
        self.check1.setChecked(True)
        self.check1.stateChanged.connect(self.on_checkbox_state_changed)  # 连接信号与槽函数

        self.check2 = QtWidgets.QCheckBox("曲线2",self.mainbox)
        self.check2.setGeometry(400, 0, 50, 20)
        self.check2.setStyleSheet("background-color: red;")
        self.check2.setChecked(True)
        self.check2.stateChanged.connect(self.on_checkbox_state_changed)  # 连接信号与槽函数
        
        self.check3 = QtWidgets.QCheckBox("曲线3",self.mainbox)
        self.check3.setGeometry(460, 0, 50, 20)
        self.check3.setStyleSheet("background-color: green;")
        self.check3.setChecked(True)
        self.check3.stateChanged.connect(self.on_checkbox_state_changed)  # 连接信号与槽函数
        
        self.check4 = QtWidgets.QCheckBox("曲线4",self.mainbox)
        self.check4.setGeometry(520, 0, 50, 20)
        self.check4.setStyleSheet("background-color: blue;")
        self.check4.setChecked(True)
        self.check4.stateChanged.connect(self.on_checkbox_state_changed)  # 连接信号与槽函数

        self.label = QtWidgets.QLabel("协议示例:   printf(\"<plots,%d,%d,%d,%d>\",p1,p2,p3,p4);   ",self.mainbox)
        self.label.setGeometry(10, 30, 400, 20)

        self.canvas = pg.GraphicsLayoutWidget(self.mainbox)
        self.canvas.setGeometry(0, 60, 800, 600)  # 设置位置和大小
初始化4个曲线参数
设置曲线最大100个数据宽度,设置4个曲线颜色

        # Line plot
        self.cpuplot = self.canvas.addPlot(title=" ", axisItems={'bottom': TimeAxisItem(orientation='bottom')})
        self.p1 = self.cpuplot.plot(pen='y')
        self.p2 = self.cpuplot.plot(pen='r')
        self.p3 = self.cpuplot.plot(pen='g')
        self.p4 = self.cpuplot.plot(pen='b')

        #### Set Data #####################
        self.X = deque(maxlen=100)
        self.Y = deque(maxlen=100)
        self.Y2 = deque(maxlen=100)
        self.Y3 = deque(maxlen=100)
        self.Y4 = deque(maxlen=100)
串口打开按键功能
打开串口并创建dataUpdate_thead线程,该线程为串口协议处理
def open_serial(self):
        #self.p1.setVisible(not self.p1.isVisible())
        if self.ser.is_open == True:
            self.ser.close()
            self.open.setText('打开串口')
            self.combex.setEnabled(True)
            self.baud.setEnabled(True)
        else:
            print(self.combex.currentText())
            self.ser = serial.Serial(self.combex.currentText(), int(self.baud.currentText()))
            if(self.ser.is_open == True):
                self.open.setText('关闭串口')
                self.combex.setEnabled(False)
                self.baud.setEnabled(False)
                self.stop = 0
                self.t = threading.Thread(target=dataUpdate_thead, args=(self,))
                self.t.daemon = True
                self.t.start()  # 线程执行
协议处理线程
协议使用的字符串形式, printf(\"<plots,%d,%d,%d,%d>\",p1,p2,p3,p4); 以此作为协议处理;
判定起始数据<,开始接收字符,未开始,则字符丢弃;判定>为结束字符,结束后开始处理内容;
将字符串通过“,”字符进行split,得到5个字符串,1个固定字符和4个数据;
判定字符串长度和校验固定字符,校验完成后将4个数据放入曲线buffer内,设置到曲线内即可;
def dataUpdate_thead(self):
    start = 0
    strs = ''
    process= []
    while True:  # 为了方便测试,让数据不停的更新
        if self.stop == 1 or self.ser.isOpen() == False:
            return
        data = self.ser.read().decode()
        if data == '<':
            start = 1
            strs = ""
        elif data == '>':
            start = 0
            split_list = strs.split(',')
            process.clear()
            for base in split_list:
                process.append( base.strip())
               
            strs = ""
            if len(process) == 5 and process[0] == "plots"  :
                current_time = int(time.time() * 1000)
                y1, y2, y3, y4 = map(int, [element.strip() for element in process[1:5]])

                self.X.append(current_time)
                self.Y.append(y1)
                self.Y2.append(y2)
                self.Y3.append(y3)
                self.Y4.append(y4)
                # 只在需要绘制时转换为数组
                self.xdata = np.array(self.X)
                self.ydata = np.array(self.Y)
                self.ydata2 = np.array(self.Y2)
                self.ydata3 = np.array(self.Y3)
                self.ydata4 = np.array(self.Y4)
                self.p1.setData(self.xdata, self.ydata)
                self.p2.setData(self.xdata, self.ydata2)
                self.p3.setData(self.xdata, self.ydata3)
                self.p4.setData(self.xdata, self.ydata4)
                if self.cnt == 2000:
                    self.cnt = 0
                    self.X = self.X[1000:]
                    self.Y = self.Y[1000:]
                    self.Y2 = self.Y2[1000:]
                    self.Y3 = self.Y3[1000:]
                    self.Y4 = self.Y4[1000:]
                self.cnt += 1
               

        elif start == 1 :
            strs +=data
以上曲线绘制软件就完成;
以下是完成后的界面;自适应范围


serial_plots.rar (2.65 KB)




708946789d01d799ef.png (39.95 KB )

708946789d01d799ef.png

使用特权

评论回复

相关帖子

发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

21

主题

114

帖子

2

粉丝