基于arduino串行输出启动python tkinter秒表

Start python tkinter stopwatch based on arduino serial output

我有一个连接到按钮的arduino。按下按钮时,串行输出 1 通过串行发送。我希望 python tkinter 秒表在按下按钮时启动。目前我知道 python 串行读取正在读取 1。并在 python 终端打印出来。但我无法控制 tkinter 秒表。 PS:我是 python 的新手。下面是我当前的代码。

from tkinter import *
import time
import serial

ser = serial.Serial(
    port='COM4',\
    baudrate=57600,\
    parity=serial.PARITY_NONE,\
    stopbits=serial.STOPBITS_ONE,\
    bytesize=serial.EIGHTBITS,\
        timeout=10)

print("connected to: " + ser.portstr)



class StopWatch(Frame):  
    """ Implements a stop watch frame widget. """                                                                
    def __init__(self, parent=None, **kw):        
        Frame.__init__(self, parent, kw)
        self._start = 0.0        
        self._elapsedtime = 0.0
        self._running = 0
        self.timestr = StringVar()               
        self.makeWidgets()      

    def makeWidgets(self):                         
        """ Make the time label. """
        l = Label(self, textvariable=self.timestr)
        self._setTime(self._elapsedtime)
        l.pack(fill=X, expand=NO, pady=2, padx=2)                      

    def _update(self): 
        """ Update the label with elapsed time. """
        self._elapsedtime = time.time() - self._start
        self._setTime(self._elapsedtime)
        self._timer = self.after(50, self._update)

    def _setTime(self, elap):
        """ Set the time string to Minutes:Seconds:Hundreths """
        minutes = int(elap/60)
        seconds = int(elap - minutes*60.0)
        hseconds = int((elap - minutes*60.0 - seconds)*100)                
        self.timestr.set('%02d:%02d:%02d' % (minutes, seconds, hseconds))

    def Start(self):                                                     
        """ Start the stopwatch, ignore if running. """
        if not self._running:            
            self._start = time.time() - self._elapsedtime
            self._update()
            self._running = 1        

    def Stop(self):                                    
        """ Stop the stopwatch, ignore if stopped. """
        if self._running:
            self.after_cancel(self._timer)            
            self._elapsedtime = time.time() - self._start    
            self._setTime(self._elapsedtime)
            self._running = 0

    def Reset(self):                                  
        """ Reset the stopwatch. """
        self._start = time.time()         
        self._elapsedtime = 0.0    
        self._setTime(self._elapsedtime)


def main():
    root = Tk()
    sw = StopWatch(root)
    sw.pack(side=TOP)


    root.mainloop()

if __name__ == '__main__':
    main()

    root = Tk()
    sw = StopWatch(root)
    sw.pack(side=TOP)
    count=1

    while True:
        for line in ser.read():

            print(chr(line))
            count = count+1
            if chr(line) == '1':
                sw.Start()

    ser.close()

编辑:根据评论回答,我已经开始工作了。下面是工作代码。

from tkinter import *
import time
import serial

class StopWatch(Frame):  
    """ Implements a stop watch frame widget. """                                                                
    def __init__(self, parent=None, **kw):        
        Frame.__init__(self, parent, kw)
        self._start = 0.0        
        self._elapsedtime = 0.0
        self._running = 0
        self.timestr = StringVar()               
        self.makeWidgets()      

    def makeWidgets(self):                         
        """ Make the time label. """
        l = Label(self, textvariable=self.timestr)
        self._setTime(self._elapsedtime)
        l.pack(fill=X, expand=NO, pady=2, padx=2)                      

    def _update(self): 
        """ Update the label with elapsed time. """
        self._elapsedtime = time.time() - self._start
        self._setTime(self._elapsedtime)
        self._timer = self.after(50, self._update)

    def _setTime(self, elap):
        """ Set the time string to Minutes:Seconds:Hundreths """
        minutes = int(elap/60)
        seconds = int(elap - minutes*60.0)
        hseconds = int((elap - minutes*60.0 - seconds)*100)                
        self.timestr.set('%02d:%02d:%02d' % (minutes, seconds, hseconds))

    def Start(self):                                                     
        """ Start the stopwatch, ignore if running. """
        if not self._running:            
            self._start = time.time() - self._elapsedtime
            self._update()
            self._running = 1        

    def Stop(self):                                    
        """ Stop the stopwatch, ignore if stopped. """
        if self._running:
            self.after_cancel(self._timer)            
            self._elapsedtime = time.time() - self._start    
            self._setTime(self._elapsedtime)
            self._running = 0

    def Reset(self):                                  
        """ Reset the stopwatch. """
        self._start = time.time()         
        self._elapsedtime = 0.0    
        self._setTime(self._elapsedtime)

def Read():
        ser = serial.Serial(
        port='COM4',\
        baudrate=57600,\
        parity=serial.PARITY_NONE,\
        stopbits=serial.STOPBITS_ONE,\
        bytesize=serial.EIGHTBITS,\
            timeout=10)

        print("connected to: " + ser.portstr)
        count=1

        while True:
                for line in ser.read():

                    print(chr(line))
                    count = count+1
                    return chr(line)

        ser.close()

def main():
    root = Tk()
    sw = StopWatch(root)
    sw.pack(side=TOP)
    ser = Read()

    if ser == '1':
        sw.Start()

    root.mainloop()

if __name__ == '__main__':
    main()

你的 main()if __name__ 语句有点破。

根据我在您的代码中看到的,您的代码不会 运行 您认为的那样。

您应该只创建一个 Tk() 的实例,并且在您的代码中它被写入了两次。

请注意,您在 root.mainloop() 之后编写的任何代码都不会 运行 直到 mainloop() 结束。届时您的程序将关闭,并且由于您的 main() 语句的其余部分,将创建一个新实例。

这可能不是您的本意。

这个:

def main():
    root = Tk()
    sw = StopWatch(root)
    sw.pack(side=TOP)


    root.mainloop()

if __name__ == '__main__':
    main()

    root = Tk()
    sw = StopWatch(root)
    sw.pack(side=TOP)
    count=1

    while True:
        for line in ser.read():

            print(chr(line))
            count = count+1
            if chr(line) == '1':
                sw.Start()

    ser.close()

大概应该是这样的:

def main():
    root = Tk()
    sw = StopWatch(root)
    sw.pack(side=TOP)

    count=1
    while True:
        for line in ser.read():
            print(chr(line))
            count = count+1
            if chr(line) == '1':
                sw.Start()

    ser.close()

    root.mainloop()

if __name__ == '__main__':
    main()

这并不是要回答您的整体问题,因为我还没有详细查看您的所有代码,但很难将它放在评论中,所以我写在这里。

更新:

我无法在任何与串行相关的内容上测试您的代码,但是我确实修改了您的 for 循环以测试您代码的功能。话虽如此,我相信您想将 ser = serial.Serial() 的内容移动到 while 循环之前的 main() 函数中。另外,也许您应该稍微更改一下 while 循环语句。目前语句 while True: 将永远 运行。

而是使用这样的东西。

x = True
while x == True:
    # do stuff
    x = False
    sw.Start()

最后我认为你的代码应该是这样的:

from tkinter import *
import time
import serial


class StopWatch(Frame):  
    """ Implements a stop watch frame widget. """                                                                
    def __init__(self, parent=None, **kw):        
        Frame.__init__(self, parent, kw)
        self._start = 0.0        
        self._elapsedtime = 0.0
        self._running = 0
        self.timestr = StringVar()               
        self.makeWidgets()      

    def makeWidgets(self):                         
        """ Make the time label. """
        l = Label(self, textvariable=self.timestr)
        self._setTime(self._elapsedtime)
        l.pack(fill=X, expand=NO, pady=2, padx=2)                      

    def _update(self): 
        """ Update the label with elapsed time. """
        self._elapsedtime = time.time() - self._start
        self._setTime(self._elapsedtime)
        self._timer = self.after(50, self._update)

    def _setTime(self, elap):
        """ Set the time string to Minutes:Seconds:Hundreths """
        minutes = int(elap/60)
        seconds = int(elap - minutes*60.0)
        hseconds = int((elap - minutes*60.0 - seconds)*100)                
        self.timestr.set('%02d:%02d:%02d' % (minutes, seconds, hseconds))

    def Start(self):                                                     
        """ Start the stopwatch, ignore if running. """
        if not self._running:            
            self._start = time.time() - self._elapsedtime
            self._update()
            self._running = 1        

    def Stop(self):                                    
        """ Stop the stopwatch, ignore if stopped. """
        if self._running:
            self.after_cancel(self._timer)            
            self._elapsedtime = time.time() - self._start    
            self._setTime(self._elapsedtime)
            self._running = 0

    def Reset(self):                                  
        """ Reset the stopwatch. """
        self._start = time.time()         
        self._elapsedtime = 0.0    
        self._setTime(self._elapsedtime)


def main():
    root = Tk()
    sw = StopWatch(root)
    sw.pack(side=TOP)
    ser = serial.Serial(
                        port='COM4',\
                        baudrate=57600,\
                        parity=serial.PARITY_NONE,\
                        stopbits=serial.STOPBITS_ONE,\
                        bytesize=serial.EIGHTBITS,\
                            timeout=10)
    x = True
    count=1
    while x == True:
        for line in ser.read():
            print(chr(line))
            count = count+1
            if chr(line) == '1':
                sw.Start()

    root.mainloop()

if __name__ == '__main__':
    main()