python串口助手

  • 2019 年 10 月 10 日
  • 筆記

  最近項目中要使用模擬數據源通過向外發送數據,以前都是用C#編寫,最近在研究python,所以就用python寫了一個串口助手,方便以後的測試。

       在電腦上通過虛擬串口助手產生兩個虛擬串口,運行編寫的串口助手

另外,打開一個串口調試助手,輔助測試

 

  兩者互發數據,可以看到編寫的串口助手能夠正常接收發送,並且在後台我們也把接收到的數據列印了出來

 

 

 

下面先講解關於介面的程式碼,這裡只是簡單的使用tkinter做介面,如果想要更好的介面效果,可以嘗試一下QT。

首先是介面中串口設置區域的程式碼,介面布局統一用grid來布局,當然也有其他的布局方法

 1 #-------------------------------------------------------------------------   2 #面板布局區   3 root = Tk()   4 root.title('xutopia 公眾號:洛水梅家')   5   6 #-------------------------------------------------------------------------   7 #串口設置區   8 label_com = Label(root, text="串口號", height=2).grid(row=0, column=0)   9 label_bps = Label(root, text="波特率", height=2).grid(column=0, row=1)  10 label_datBit = Label(root, text="數據位", height=2).grid(column=0, row=2)  11 label_parity = Label(root, text="校驗位", height=2).grid(column=0, row=3)  12 label_stop_bit = Label(root, text="停止位", height=2).grid(column=0, row=4)

接下來是串口號的combobox的設置,其中serialPortFile.GetCom()是獲取所有串口號的函數,接下來會講解

#串口號  varPort = StringVar()  combo_com = ttk.Combobox(root, textvariable=varPort, width=8, height=2, justify=CENTER)  serial_com = serialPortFile.GetCom()  combo_com['values'] = serial_com  #combo_com.bind("<<ComboboxSelected>>", lambda event: combo1_handler(var=varPort.get()))  combo_com.current(0)  combo_com.grid(column=1, row=0)

 1 #波特率   2 varBitrate = StringVar()   3 combo_bps = ttk.Combobox(root, textvariable=varBitrate, width=8, height=2, justify=CENTER)   4 combo_bps['values'] = ("9600", "19200", "38400", "115200")   5 #combo_bps.bind("<<ComboboxSelected>>", lambda event: combo2_handler(var=varBitrate.get()))   6 combo_bps.current(0)   7 combo_bps.grid(column=1, row=1)   8 #數據位   9 combo_byteBit = ttk.Combobox(root, width=8, height=2, justify=CENTER)  10 combo_byteBit['values'] = ("5", "6", "7", "8")  11 combo_byteBit.current(3)  12 combo_byteBit.grid(column=1, row=2)  13 #奇偶校驗  14 combo_parity = ttk.Combobox(root, width=8, height=2, justify=CENTER)  15 combo_parity['values'] = ("N", "O", "E")  16 combo_parity.current(0)  17 combo_parity.grid(column=1, row=3)  18 #停止位  19 combo_stopBit = ttk.Combobox(root, width=8, height=2, justify=CENTER)  20 combo_stopBit['values'] = ("1", "1.5", "2")  21 combo_stopBit.current(0)  22 combo_stopBit.grid(column=1, row=4)  23 serialPortFile.text_rx = Text(root,width=70,height=20)  24 serialPortFile.text_rx.grid(row=0,column=3,rowspan=5)  25 serialPortFile.text_rx.insert(END,'這是一個xutopia用python編寫的串口助手,公眾號:洛水梅家n')  26 text_tx = Text(root,width=70,height=10)  27 text_tx.grid(row=5,column=3,rowspan=2)  28 text_tx.insert(END,'xutopia 公眾號:洛水梅家,發送數據12345,上山打老虎')  29 button_send = Button(root, text='send', width=18, height=1)  30 button_send.bind("<Button-1>", lambda event: serialPortFile.usart_sent(var=text_tx.get("0.0", "end")))  31 button_send.grid(column=1, row=6)  32 #串口開關按鈕  33 serialPortFile.button_var = StringVar()  34 serialPortFile.button_var.set("打開串口")  35 buttonOpenCom = Button(root, textvariable=serialPortFile.button_var, width=18, height=1)  36 buttonOpenCom.bind("<Button-1>", lambda event: serialPortFile.usart_ctrl(combo_com.get(), combo_bps.get(),combo_parity.get(),combo_stopBit.get(),combo_byteBit.get()))  37 buttonOpenCom.grid(column=1, row=5)

以上就是介面相關的程式碼,最終效果也就是上圖顯示的效果,很簡單。接下來介紹python串口的邏輯程式碼部分。在另外一個模組中serialPortFile.py

獲取電腦中所有的串口號的函數,注意這裡通過serial.tools.list_ports.comports()獲取所有的串口號之後進行了一步轉換,存在一個數組中,這樣,傳入serial中才能被識別。

 1 # 獲取並存儲串口號到數組   2 def GetCom():   3     port_list = list(serial.tools.list_ports.comports())   4     print(len(port_list))   5     portcnt = len(port_list)   6     serial_com = []   7     for m in range(portcnt):   8         port_list_1 = list(port_list[m])   9         serial_com.append(port_list_1[0])  10     return serial_com

串口打開關閉函數,串口打開之後創建了一個執行緒threading,執行緒中一直監視串口的狀態,一旦接受到數據,就把數據顯示到接受數據顯示框中。有關python執行緒threading的內容可以在我的公眾號,洛水梅家中查看。

注意在程式碼中加了一個if ser.is_open:的條件判斷,不加入這個條件的話,很有可能重複打開串口而報錯誤。

同時在按鍵Button的狀態也隨著串口的開關狀態而變化。

def usart_ctrl(com, bps,parity_,stopbits_,bytesize_):      #print(__file__, sys._getframe().f_lineno, port_, bitrate_, var.get())      global ser, button_var      if button_var.get() == "打開串口":          button_var.set("關閉串口")          ser = serial.Serial(              port=com,              baudrate=int(bps),              parity=parity_,              timeout=0.2,              stopbits=float(stopbits_),              bytesize=int(bytesize_))          if ser.is_open:              pass          else:              ser.open()          recv_data = threading.Thread(target=thread_recv)          recv_data.start()      else:          button_var.set("打開串口")          if ser.is_open:              ser.close()          else:              pass

def usart_sent(var):      #print(__file__, sys._getframe().f_lineno, "-->", var)      print(var)      if ser.is_open:          ser.write(var.encode("gb2312"))  def thread_recv():      global text_rx      while True:          try:              read = ser.readall()              if len(read) > 0:                  print(bytes(read).decode('gb2312'))                  # print(__file__, sys._getframe().f_lineno, "<--", bytes(read).decode('ascii'))                  text_rx.insert(END,bytes(read).decode('gb2312'))          except Exception as e:              print(e)              time.sleep(1)              pass

上面就是關於串口助手的所有程式碼介紹,更多詳細內容,關注公眾號 洛水梅家 免費獲取源碼

後台回復python_com,免費獲取源碼