文章目录
一、前言
在物联网行业摸爬滚了四年,从移动开发、嵌入式开发、云平台开发到微信小程序公众号开发,似乎已经有了整体的流程的深度认识,但这半年似乎又没有新的输出,还在啃老本,我大胆又学了一门语言Python,对于我这种有面向对象编程基础的人来说,上手非常简单,但按照我一贯的学习方法:
运筹帷幄之中,决胜千里之外。
先要规划自己要实现什么功能的软件,再去针对性地翻资料,难免怕坑难免等原因,可以投资买视频学,但是我也从来不指望任何人免费能带我,能有稍些的指点我便心满意足,别指望他人能给予你什么,而是你要以一物换一物去等价交换,终究会明白,别人的屋檐再大,都不如自己有把伞。
所以,大家一个点赞/转发我的开源文章,我非常感恩。每一个不曾学习的日子,都是对生命的辜负,但行好事,莫问前程;
所以,我也开源我上月春节假期学习到的PyQt5
,并且基于PyQt5
打造的一款跨平台的电脑串口调试助手(支持window
、MacOS
和 Linux
),代码和安装包在文章末尾开源地址;
集成了市面上大多串口工具的功能,包括:
- 自定义波特率
- 自动换行
- RTS 控
- DTR 控
- Hex 发送和接收
二、开发的必备工具
- Python 3.8 环境
- PyCharm 2020.1.1 x64
- QT Designer 5.2
对于如何选择一款开发工具,我也是挑了蛮久的,还是选择熟悉强大的IDEA系列软件,再配合QT Designer
使用起来,非常Nice。下面分享下,我的快捷键,能使你的开发效率加倍。
2.1 PyCharm 如何集成 QT Designer
UI代码转可视化
- 终端:
Python38-32\Lib\site-packages\pyqt5_tools\Qt\bin\designer.exe
可视转化UI代码
- 终端:
Python38-32\Lib\site-packages\pyqt5_tools\Qt\bin\designer.exe
- 参数:
-m PyQt5.uic.pyuic $FileName$ -o $FileNameWithoutExtension$.py
打包成 exe 软件:
- 终端:
D:\Python\QT5\Demo\Scripts\pyinstaller.exe
- 参数:
-F -w $FileName$
2.2、引进自定义控件
注意头文件引入路径,以及其父类;
移除此控件为自定义控件
三、代码结构说明
3.1 串口驱动应用层封装
- 包括串口异常中断处理;
- 提供上层应用接口有配置、发送和接收;
- 子线程和信号槽应用实现无缝读取串口通讯接收到的数据给上位机;
class UartRecieveThread(QThread):
def __init__(self, run):
super(UartRecieveThread, self).__init__()
self.runfun = run
def run(self):
self.runfun()
class UartSerial(QObject):
# 定义一个信号变量,1个参数
signalRecieve = pyqtSignal(object)
CODE_RECIEVE, CODE_DISCONNECT = 0, 1
def __init__(self):
super(UartSerial, self).__init__()
self.mThread = UartRecieveThread(self.data_receive) # 创建线程
self.mSerial = serial.Serial()
self.data_num_received = 0
def init(self, _port="/dev/ttyUSB0", _baudrate=115200, _bytesize=8, _stopbits="N", _parity=serial.PARITY_NONE):
self.mSerial.port = _port
self.mSerial.baudrate = _baudrate
self.mSerial.bytesize = _bytesize
self.mSerial.stopbits = _stopbits
self.mSerial.parity = _parity
self.data_num_received = 0
def is_port_open(self, _port_name, _baudrate):
try:
self.mSerial.port = _port_name
self.mSerial.baudrate = _baudrate
return not self.mSerial.isOpen()
except Exception:
return False
# 关闭串口
def port_close(self, _port_name, _baudrate):
self.mSerial.port = _port_name
self.mSerial.baudrate = _baudrate
try:
self.mSerial.close()
except:
return False
self.mThread.quit()
return True
# 串口检测
def get_all_port(self):
# 检测所有存在的串口,将信息存储在字典中
self.port_list_name = []
port_list = list(serial.tools.list_ports.comports())
i = 0
if len(port_list) <= 0:
return []
else:
for port in port_list:
i = i + 1
self.port_list_name.append(port[0])
return self.port_list_name
# 打开串口
def try_port_open(self, _port, _baudrate=115200):
self.mSerial.port = _port
self.mSerial.baudrate = _baudrate
try:
self.mSerial.open()
except:
return False
if not self.mThread.isRunning():
self.mThread.start()
return True
def set_rts(self, IsTrue):
self.mSerial.setRTS(IsTrue)
def set_dts(self, IsTrue):
self.mSerial.setDTR(IsTrue)
def get_rts(self):
return self.mSerial.rts
def get_dts(self):
return self.mSerial.dtr
def set_bytesize(self, _bytesize):
self.mSerial.bytesize = _bytesize
def set_parity(self, _parity):
self.mSerial.parity = _parity
def set_stopbits(self, _stopbits):
self.mSerial.stopbits = _stopbits
# 接收数据
def data_receive(self):
while True:
data = {}
try:
num = self.mSerial.inWaiting()
except:
self.mSerial.close()
data['code'] = self.CODE_DISCONNECT
data['data'] = 0
data['length'] = 0
self.signalRecieve.emit(data)
return None
if num > 0:
if self.mSerial.isOpen():
buff = self.mSerial.read(num)
data['code'] = self.CODE_RECIEVE
data['data'] = buff
data['length'] = len(buff)
self.signalRecieve.emit(data)
time.sleep(0.1)
# 设置回调函数
def setCallBack(self, funtion):
self.signalRecieve.connect(funtion)
# send data
def send_data(self, buff="", isHexSend=False, _port="", _baudrate=115200):
if buff != "":
# 非空字符串
if isHexSend:
# hex发送
buff = buff.strip()
send_list = []
while buff != '':
try:
num = int(buff[0:2], 16)
except ValueError:
return None
buff = buff[2:].strip()
send_list.append(num)
buff = bytes(send_list)
num = self.mSerial.write(buff)
3.2 自定义控件
在实现某些功能时候,发现自带的控件无法满足需求时候,我们只能自定义控件,我重写了几个控件:
1 、重写 QComboBox
为了实现能自定义波特率,比如用户输入74880;
class MyQComBox(QComboBox):
popupAboutToBeShown = pyqtSignal() # 创建一个信号
def showPopup(self): # 重写showPopup函数
super(MyQComBox, self).showPopup()
self.popupAboutToBeShown.emit() # 发送信号
2、重写 QWidget
为了能够全局监听键盘按键,实现一键清空接受栏;
class MineWidget(QWidget):
# 定义一个信号变量,1个参数
signalMineWidget = pyqtSignal(object)
def set_connect_key_press(self, fun):
self.signalMineWidget.connect(fun)
# 检测键盘回车按键
def keyPressEvent(self, event):
self.signalMineWidget.emit(event.key())
def mousePressEvent(self, event):
self.signalMineWidget.emit(event.button())
3.3 接收数据并显示
这里要特别说明的是,在解码时候,特要指明 utf-8
编码,以及带上 ignore
表示接受解码失败异常;
def at_callback_handler(obj):
if obj['code'] == 1:
if ui.bt_open_off_port.text() != '打开串口':
QMessageBox.critical(MainWindow, '错误信息', '串口异常断开!')
# 面板可操作
checkoutPortStatus(True)
ui.bt_open_off_port.setText('打开串口')
refreshPort()
else:
buff = (obj['data'])
now_time = datetime.now() # 获取当前时间
# new_time = now_time.strftime('[%H:%M:%S]') # 打印需要的信息,依次是年月日,时分秒,注意字母大小写
if ui.checkBox_show_hex.checkState():
out_s = ''
for i in range(0, len(buff)):
out_s = out_s + '{:02X}'.format(buff[i]) + ' '
if ui.checkBox_show_add_ctrl.checkState():
ui.textBrowserShow.append(out_s)
else:
ui.textBrowserShow.insertPlainText(out_s)
ui.textBrowserShow.moveCursor(QTextCursor.End)
else:
try:
if ui.checkBox_show_add_ctrl.checkState():
ui.textBrowserShow.append(buff.decode('utf-8', 'ignore'))
else:
ui.textBrowserShow.insertPlainText(buff.decode('utf-8', 'ignore'))
ui.textBrowserShow.moveCursor(QTextCursor.End)
except:
# 乱码显示
pass
四、效果
关注微信公众号 徐宏blog 发送 210317 以获取全部代码
另外,不要把我的博客作为学习标准,我的只是笔记,难有疏忽之处,如果有,请指出来,也欢迎留言哈!
- 玩转
esp8266
带你飞、加群QQ
群,不喜的朋友勿喷勿加:434878850 - 个人邮箱:xuhongv@yeah.net 24小时在线,有发必回复!
- esp8266源代码学习汇总(持续更新,欢迎star):https://github.com/xuhongv/StudyInEsp8266
- esp32源代码学习汇总(持续更新,欢迎star):https://github.com/xuhongv/StudyInEsp32
- 关注下面微信公众号二维码,干货多多,第一时间推送!