使用Python串口实时显示数据并绘图的例子


Posted in Python onDecember 26, 2019

使用pyserial进行串口传输

一、安装pyserial以及基本用法

在cmd下输入命令pip install pyserial

注:升级pip后会出现 "‘E:\Anaconda3\Scripts\pip-script.py' is not present."错误

使用 easy_install pip命令就能解决,换一条重新能执行安装的命令

常用方法:

ser = serial.Serial(0) 是打开第一个串口

print ser.portstr 能看到第一个串口的标识,windows下是COM1

ser.write(“hello") 就是往串口里面写数据

ser.close() 就是关闭ser表示的串口

ser.open() 会打开这个串口

ser = serial.Serial(‘COM1', 115200) 来设置波特率,当然还有专门的函数

data = ser.read()可以读一个字符

data = ser.read(20) 是读20个字符

data = ser.readline() 是读一行,以/n结束,要是没有/n就一直读,阻塞。

data = ser.readlines()和ser.xreadlines()都需要设置超时时间

ser.baudrate = 9600 设置波特率

ser 来查看当前串口的状态

ser.isOpen() 看看这个串口是否已经被打开

串行口的属性:

name:设备名字 portstr:已废弃,用name代替 port:读或者写端口 baudrate:波特率

bytesize:字节大小 parity:校验位 stopbits:停止位 timeout:读超时设置

writeTimeout:写超时 xonxoff:软件流控 rtscts:硬件流控 dsrdtr:硬件流控

interCharTimeout:字符间隔超时

二、最基本的串口代码

import serial
portx="COM5"
bps=9600
timex=5
#串口执行到这已经打开 再用open命令会报错
ser = serial.Serial(portx, int(bps), timeout=1, parity=serial.PARITY_NONE,stopbits=1)
if (ser.isOpen()):
 print("open success")
 # 向端口些数据 字符串必须译码
 ser.write("hello".encode()) 
 while (True):
 line = ser.readline() 
 if(line):
  print(line)
  line=0
else:
 print("open failed")
ser.close()#关闭端口
)

三、pyqtgraph的使用

pip install pyqtgraph#显示波形的界面

pip install PyQt5#界面要Qt的支持

pyqtgraph是Python平台上一种功能强大的2D/3D绘图库,相对于matplotlib库,由于内部实现方式上,使用了高速计算的numpy信号处理库以及Qt的GraphicsView框架,因此,它在大数据量的数字处理和快速显示方面有着巨大的优势,它适合于需要快速绘图更新、视频或实时交互性的操作场合。另外,它不仅为各种数据提供了快速可交互式的图形显示,同时也提供了用于快速开发应用程序的各种小工具,如属性树、流程图等小部件,在数学、科学和工程领域都有着广泛的应用。

import pyqtgraph as pg
import numpy as np
import array

app = pg.mkQApp()#建立app
win = pg.GraphicsWindow()#建立窗口
win.setWindowTitle(u'pyqtgraph逐点画波形图')
win.resize(800, 500)#小窗口大小

data = array.array('d') #可动态改变数组的大小,double型数组
historyLength = 100#横坐标长度
p = win.addPlot()#把图p加入到窗口中
p.showGrid(x=True, y=True)#把X和Y的表格打开
p.setRange(xRange=[0,historyLength], yRange=[-1.2, 1.2], padding=0)
p.setLabel(axis='left', text='y / V')#靠左
p.setLabel(axis='bottom', text='x / point')
p.setTitle('y = sin(x)')#表格的名字
curve = p.plot()#绘制一个图形
idx = 0
def plotData():
 global idx#内部作用域想改变外部域变量
 tmp = np.sin(np.pi / 50 * idx)
 if len(data)<historyLength:
 data.append(tmp)
 else:
 data[:-1] = data[1:]#前移
 data[-1] = tmp
 curve.setData(data)
 idx += 1

timer = pg.QtCore.QTimer()
timer.timeout.connect(plotData)#定时调用plotData函数
timer.start(50)#多少ms调用一次

app.exec_()

使用Python串口实时显示数据并绘图的例子

四、通过多线程实现串口数据的实时绘图import pyqtgraph as pg

主要是开了一个线程去处理串口 剩下的和上面内容一样 就不过多解释了 直接上代码

import array
import serial
import threading
import numpy as np
import time


i = 0
def Serial():
 while(True):
 n = mSerial.inWaiting()
 if(n):
  if data!=" ":
  dat = int.from_bytes(mSerial.readline(1),byteorder='little') # 格式转换
  n=0
  global i;
  if i < historyLength:
   data[i] = dat
   i = i+1
  else:
   data[:-1] = data[1:]
   data[i-1] = dat

def plotData():
 curve.setData(data)


if __name__ == "__main__":
 app = pg.mkQApp() # 建立app
 win = pg.GraphicsWindow() # 建立窗口
 win.setWindowTitle(u'pyqtgraph逐点画波形图')
 win.resize(800, 500) # 小窗口大小
 data = array.array('i') # 可动态改变数组的大小,double型数组
 historyLength = 200 # 横坐标长度
 a = 0
 data=np.zeros(historyLength).__array__('d')#把数组长度定下来
 p = win.addPlot() # 把图p加入到窗口中
 p.showGrid(x=True, y=True) # 把X和Y的表格打开
 p.setRange(xRange=[0, historyLength], yRange=[0, 255], padding=0)
 p.setLabel(axis='left', text='y / V') # 靠左
 p.setLabel(axis='bottom', text='x / point')
 p.setTitle('semg') # 表格的名字
 curve = p.plot() # 绘制一个图形
 curve.setData(data)
 portx = 'COM24'
 bps = 19200
 # 串口执行到这已经打开 再用open命令会报错
 mSerial = serial.Serial(portx, int(bps))
 if (mSerial.isOpen()):
 print("open success")
 mSerial.write("hello".encode()) # 向端口些数据 字符串必须译码
 mSerial.flushInput() # 清空缓冲区
 else:
 print("open failed")
 serial.close() # 关闭端口
 th1 = threading.Thread(target=Serial)#目标函数一定不能带()被这个BUG搞了好久
 th1.start()
 timer = pg.QtCore.QTimer()
 timer.timeout.connect(plotData) # 定时刷新数据显示
 timer.start(50) # 多少ms调用一次
 app.exec_()

效果如图

使用Python串口实时显示数据并绘图的例子

五、与下位机通讯实现波形实时监测

在这里与第四阶段基本相同,需要注意的是,如果收数据直接画图的话,波形会出现问题。所以串口传输数据时使用循环队列(先进先出),数据来之后先进队列,之后再定时器调用函数,出队列,更新图。理论上刷新数据的时间需要大于下位机发送数据的间隔时间,否则队列会越来越大,而且图的刷新不连贯。再就是有一个小问题,因为正弦波有负值,我又没找到很好的把Byte转为char的方法,所以只能手动代码处理,先转成int类型,再把第八位(符号位)清零,得到绝对值。然后再取负,得到我们需要的数据。但发现Python无法进行移位操作,python是int类型是无精度类型,不会发生溢出而进行截取的情况,所以只能先转为二进制在移位,太麻烦,直接通过减去一个数的方法来实现了。然后直接上代码吧

import pyqtgraph as pg
import array
import serial
import threading
import numpy as np
from queue import Queue
import time


i = 0
q = Queue(maxsize=0)
def Serial():
 global i;
 global q;
 while(True):
 n = mSerial.inWaiting()
 if(n):
  dat = int.from_bytes(mSerial.readline(1),byteorder='little') # 格式转换
  if(dat>>7):
  dat =256-dat
  dat =0-dat
  q.put(dat)

def plotData():
 global i;
 if i < historyLength:
 data[i] = q.get()
 i = i+1
 else:
 data[:-1] = data[1:]
 data[i-1] = q.get()
 curve.setData(data)


if __name__ == "__main__":
 app = pg.mkQApp() # 建立app
 win = pg.GraphicsWindow() # 建立窗口
 win.setWindowTitle(u'pyqtgraph逐点画波形图')
 win.resize(800, 500) # 小窗口大小
 data = array.array('i') # 可动态改变数组的大小,double型数组
 historyLength = 100 # 横坐标长度
 a = 0
 data=np.zeros(historyLength).__array__('d')#把数组长度定下来
 p = win.addPlot() # 把图p加入到窗口中
 p.showGrid(x=True, y=True) # 把X和Y的表格打开
 p.setRange(xRange=[0, historyLength], yRange=[-50, 50], padding=0)
 p.setLabel(axis='left', text='y / V') # 靠左
 p.setLabel(axis='bottom', text='x / point')
 p.setTitle('semg') # 表格的名字
 curve = p.plot() # 绘制一个图形
 curve.setData(data)
 portx = 'COM25'
 bps = 19200
 # 串口执行到这已经打开 再用open命令会报错
 mSerial = serial.Serial(portx, int(bps))
 if (mSerial.isOpen()):
 dat = 0xff;
 dat >> 2;
 print("open success")
 # 向端口些数据 字符串必须译码
 mSerial.write("hello".encode())
 mSerial.flushInput() # 清空缓冲区
 else:
 print("open failed")
 serial.close() # 关闭端口
 th1 = threading.Thread(target=Serial)
 th1.start()
 timer = pg.QtCore.QTimer()
 timer.timeout.connect(plotData) # 定时刷新数据显示
 timer.start(1) # 多少ms调用一次
 app.exec_()

以上这篇使用Python串口实时显示数据并绘图的例子就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
浅谈Python中copy()方法的使用
May 21 Python
解析Python中的生成器及其与迭代器的差异
Jun 20 Python
对python修改xml文件的节点值方法详解
Dec 24 Python
python中logging模块的一些简单用法的使用
Feb 22 Python
windows系统中Python多版本与jupyter notebook使用虚拟环境的过程
May 15 Python
pytorch制作自己的LMDB数据操作示例
Dec 18 Python
浅谈对pytroch中torch.autograd.backward的思考
Dec 27 Python
python实现对变位词的判断方法
Apr 05 Python
使用python检查yaml配置文件是否符合要求
Apr 09 Python
Keras官方中文文档:性能评估Metrices详解
Jun 15 Python
Django+RestFramework API接口及接口文档并返回json数据操作
Jul 12 Python
python 自定义异常和主动抛出异常(raise)的操作
Dec 11 Python
简单了解python字符串前面加r,u的含义
Dec 26 #Python
python关于调用函数外的变量实例
Dec 26 #Python
python3 实现调用串口功能
Dec 26 #Python
将python依赖包打包成window下可执行文件bat方式
Dec 26 #Python
python小项目之五子棋游戏
Dec 26 #Python
python自动识别文本编码格式代码
Dec 26 #Python
Python基于pygame实现单机版五子棋对战
Dec 26 #Python
You might like
PHP的Json中文处理解决方案
2016/09/29 PHP
php面向对象之反射功能与用法分析
2017/03/29 PHP
使Ext的Template可以解析二层的json数据的方法
2007/12/22 Javascript
javascript 异常处理使用总结
2009/06/21 Javascript
Prototype源码浅析 String部分(一)之有关indexOf优化
2012/01/15 Javascript
基于jQuery的图片左右无缝滚动插件
2012/05/23 Javascript
jquery禁用右键单击功能屏蔽F5刷新
2014/03/17 Javascript
用js提交表单解决一个页面有多个提交按钮的问题
2014/09/01 Javascript
javascript 构造函数方式定义对象
2015/01/02 Javascript
jquery通过load获取文件的内容并跳到锚点的方法
2015/01/29 Javascript
理解JavaScript中worker事件api
2015/12/25 Javascript
seajs模块之间依赖的加载以及模块的执行
2016/10/21 Javascript
如何防止INPUT按回车自动提交表单FORM
2016/12/06 Javascript
Bootstrap中glyphicons-halflings-regular.woff字体报404错notfound的解决方法
2017/01/19 Javascript
在vue项目中使用Nprogress.js进度条的方法
2018/01/31 Javascript
使用jQuery mobile NuGet让你的网站在移动设备上同样精彩
2019/06/18 jQuery
JavaScript经典案例之简易计算器
2020/08/24 Javascript
400多行Python代码实现了一个FTP服务器
2012/05/10 Python
asyncio 的 coroutine对象 与 Future对象使用指南
2016/09/11 Python
python利用smtplib实现QQ邮箱发送邮件
2020/05/20 Python
Python3实现将本地JSON大数据文件写入MySQL数据库的方法
2018/06/13 Python
对python中类的继承与方法重写介绍
2019/01/20 Python
python topk()函数求最大和最小值实例
2020/04/02 Python
Pycharm 如何一键加引号的方法步骤
2021/02/05 Python
Footshop罗马尼亚:最好的运动鞋选择
2019/09/10 全球购物
计算机专业毕业生的自我评价
2013/11/18 职场文书
电子商务专业求职信
2014/03/08 职场文书
2014年新生军训方案
2014/05/01 职场文书
党员领导干部承诺书
2014/05/28 职场文书
房屋登记授权委托书范本
2014/10/09 职场文书
教师工作表现评语
2014/12/31 职场文书
学校工会工作总结2015
2015/05/19 职场文书
2015年数学教研工作总结
2015/07/22 职场文书
文案策划岗位个人自我评价(范文)
2019/08/08 职场文书
python - timeit 时间模块
2021/04/06 Python
HTML静态页面获取url参数和UserAgent的实现
2022/08/05 HTML / CSS