串口通信是嵌入式工程师的必备技能,如果能自己写一个简单的上位机的话,肯定会加分不少。
本系列是把相关的工作做个记录,毕竟Python不是经常用,老是忘记很多东西。用的时候总是重头再学一遍太麻烦了。毕竟3个月以后,自己写的程序也如同天书一般。
开发环境:
Python版本是最新的3.10.1,
编译和调试用的是PyCharm2021.4社区版,
需要用到的库:串口通信库pyserial,命令行:pip3 install pyserial回车即可安装。
串口设备:自己的STM32板子
各种版本的软件,工具,我都喜欢用新版本,主要工具软件 大概是每年更新一次。
因为之前吃过亏,Protel99用了好多年,后面一下用到AD2013,感觉全变了,极不适应。还有UG也是这样。每年升级的话,软件更新不会太多,接受起来比较容易。
首先导入pyserial库和定义全局变量
import serial import serial.tools.list_ports import threading com_rx_buf = '' # 接收缓冲区 com_tx_buf = '' # 发送缓冲区 COMM = serial.Serial() # 定义串口对象 port_list: list # 可用串口列表 port_select: list # 选择好的串口
port_list = serial.tools.list_ports.comports() # 返回值为可用的串口列表,其__len__属性为可用串口的数量。本机的port_list.len()为2。
post_list.len(),为2
post_list[0].device字段为串口编号,为COM99
post_list[1].device字段为串口编号,为COM97
串口的编号,在设备管理器中是可以设置的,我一般是将指定设备安排一个比较大的串口号,列表的最后一项就是我想要的设备了。
COMM = serial.Serial(serial_port, 115200, timeout=0.01) if COMM.isOpen(): print(serial_port, "open success") return 0 else: print("open failed") return 255
对应的是关闭串口,程序结束时必须关闭串口,否则,如果程序异常退出后,再次运行程序时,串口会因为已经处于打开状态而出错。
COMM.close()
COMM.write(buf)
buf = COMM.readall() 可以读取出串口接收的所有数据。可以开一个接收线程,里面是一个死循环,每经过一定的延迟时间,执行一次本指令即可。
需要注意的是,如果对方发送的是一长串数据,又恰好在数据发送了一部分的时候延时时间到,这样收到的数据就不完全,此时可以再经过一个延迟,再执行一次readall(),即可读取全部数据了。这个工作放在接收线程中完成即可。
# 安装:pip3 install pyserial //python3 import serial import serial.tools.list_ports import time import threading com_rx_buf = '' # 接收缓冲区 com_tx_buf = '' # 发送缓冲区 COMM = serial.Serial() # 定义串口对象 port_list: list # 可用串口列表 port_select: list # 选择好的串口 # 无串口返回0, # 返回可用的串口列表 def get_com_list(): global port_list # a = serial.tools.list_ports.comports() # print(a) # port_list = list(serial.tools.list_ports.comports()) port_list = serial.tools.list_ports.comports() return port_list def set_com_port(n=0): global port_list global port_select port_select = port_list[n] return port_select.device # 打开串口 def serial_open(n=0): global COMM serial_port = set_com_port(n) COMM = serial.Serial(serial_port, 115200, timeout=0.01) if COMM.isOpen(): print(serial_port, "open success") return 0 else: print("open failed") return 255 # 关闭串口 def serial_close(): global COMM COMM.close() print(COMM.name + "closed.") def set_com_rx_buf(buf=''): global com_rx_buf com_rx_buf = buf def set_com_tx_buf(buf=''): global com_tx_buf com_tx_buf = buf def get_com_rx_buf(): global com_rx_buf return com_rx_buf def get_com_tx_buf(): global com_tx_buf return com_tx_buf def thread_com_receive(): while True: try: rx_buf = '' rx_buf = COMM.read() # 转化为整型数字 if rx_buf != b'': time.sleep(0.01) rx_buf = rx_buf + COMM.read_all() print("串口收到消息:", rx_buf) time.sleep(0.01) except: pass pass # def serial_encode(addr=0, command=0, param1=0, param0=0): # buf = [addr, command, param1, param0, 0, 0, 0, 0] # print(buf) # return buf def serial_send_command(addr=0, command=0, param1=0, param0=0, data3=0, data2=0, data1=0, data0=0): buf = [addr, command, param1, param0, data3, data2, data1, data0] COMM.write(buf) pass def serial_init(): buf = "AT+CG\r\n" COMM.write(buf) time.sleep(0.05) buf = COMM.read_all() if buf != "OK\r\n": return 254 # 进入调试模式失败 buf = "AT+CAN_MODE=0\r\n" COMM.write(buf) time.sleep(0.05) buf = COMM.read_all() if buf != "OK\r\n": return 253 # 进入正常模式失败,模块处于1状态,即环回模式中 buf = "AT+CAN_BAUD=500000\r\n" COMM.write(buf) time.sleep(0.05) buf = COMM.read_all() if buf != "OK\r\n": return 253 # 波特率设置失败 buf = "AT+FRAMEFORMAT=1,0,\r\n" COMM.write(buf) time.sleep(0.05) buf = COMM.read_all() if buf != "OK\r\n": return 253 # 波特率设置失败 buf = "AT+ET\r\n" # 进入透传模式 COMM.write(buf) time.sleep(0.05) buf = COMM.read_all() if buf != "OK\r\n": return 255 # 不是CAN模块 if __name__ == '__main__': get_com_list() len = port_list.__len__() device = port_list[0].device print(len, device) serial_open() thread1 = threading.Thread(target=thread_com_receive) thread1.start() # serial_close()