116

0.5秒ごとに機能を起動し、タイマーを開始、停止、リセットできるようにしたい。私はPythonスレッドがどのように機能するかについてあまり知識がなく、Pythonタイマーに問題があります。

しかし、 2回RuntimeError: threads can only be started once実行すると取得し続けます。threading.timer.start()これに対する回避策はありますか?threading.timer.cancel()毎回始める前に応募してみました。

擬似コード:

t=threading.timer(0.5,function)
while True:
    t.cancel()
    t.start()
4

14 に答える 14

133

最良の方法は、タイマースレッドを1回開始することです。タイマースレッド内で、次のようにコーディングします

class MyThread(Thread):
    def __init__(self, event):
        Thread.__init__(self)
        self.stopped = event

    def run(self):
        while not self.stopped.wait(0.5):
            print("my thread")
            # call a function

タイマーを開始したコードでset、stoppedイベントを使用してタイマーを停止できます。

stopFlag = Event()
thread = MyThread(stopFlag)
thread.start()
# this will stop the timer
stopFlag.set()
于 2012-09-15T06:56:41.490 に答える
37

PythonのsetIntervalと同等のものから:

import threading

def setInterval(interval):
    def decorator(function):
        def wrapper(*args, **kwargs):
            stopped = threading.Event()

            def loop(): # executed in another thread
                while not stopped.wait(interval): # until stopped
                    function(*args, **kwargs)

            t = threading.Thread(target=loop)
            t.daemon = True # stop if the program exits
            t.start()
            return stopped
        return wrapper
    return decorator

使用法:

@setInterval(.5)
def function():
    "..."

stop = function() # start timer, the first call is in .5 seconds
stop.set() # stop the loop
stop = function() # start new timer
# ...
stop.set() 

または、これは同じ機能ですが、デコレータの代わりにスタンドアロン関数として使用されます

cancel_future_calls = call_repeatedly(60, print, "Hello, World")
# ...
cancel_future_calls() 

スレッドを使用せずにそれを行う方法は次のとおりです。

于 2013-05-03T22:40:03.147 に答える
34

Hans Thenの答えを少し改善すると、Timer関数をサブクラス化できます。以下は、 「リピートタイマー」コード全体になり、すべて同じ引数を持つthreading.Timerのドロップイン置換として使用できます。

from threading import Timer

class RepeatTimer(Timer):
    def run(self):
        while not self.finished.wait(self.interval):
            self.function(*self.args, **self.kwargs)

使用例:

def dummyfn(msg="foo"):
    print(msg)

timer = RepeatTimer(1, dummyfn)
timer.start()
time.sleep(5)
timer.cancel()

次の出力を生成します。

foo
foo
foo
foo

timer = RepeatTimer(1, dummyfn, args=("bar",))
timer.start()
time.sleep(5)
timer.cancel()

を生成します

bar
bar
bar
bar
于 2018-02-12T06:57:11.033 に答える
33

タイマースレッドの使用-

from threading import Timer,Thread,Event


class perpetualTimer():

   def __init__(self,t,hFunction):
      self.t=t
      self.hFunction = hFunction
      self.thread = Timer(self.t,self.handle_function)

   def handle_function(self):
      self.hFunction()
      self.thread = Timer(self.t,self.handle_function)
      self.thread.start()

   def start(self):
      self.thread.start()

   def cancel(self):
      self.thread.cancel()

def printer():
    print 'ipsem lorem'

t = perpetualTimer(5,printer)
t.start()

これはによって停止することができますt.cancel()

于 2014-06-30T10:29:32.150 に答える
17

OPが要求したようにタイマーを使用して正しい答えを提供するために、swapniljariwalaの答えを改善します。

from threading import Timer


class InfiniteTimer():
    """A Timer class that does not stop, unless you want it to."""

    def __init__(self, seconds, target):
        self._should_continue = False
        self.is_running = False
        self.seconds = seconds
        self.target = target
        self.thread = None

    def _handle_target(self):
        self.is_running = True
        self.target()
        self.is_running = False
        self._start_timer()

    def _start_timer(self):
        if self._should_continue: # Code could have been running when cancel was called.
            self.thread = Timer(self.seconds, self._handle_target)
            self.thread.start()

    def start(self):
        if not self._should_continue and not self.is_running:
            self._should_continue = True
            self._start_timer()
        else:
            print("Timer already started or running, please wait if you're restarting.")

    def cancel(self):
        if self.thread is not None:
            self._should_continue = False # Just in case thread is running and cancel fails.
            self.thread.cancel()
        else:
            print("Timer never started or failed to initialize.")


def tick():
    print('ipsem lorem')

# Example Usage
t = InfiniteTimer(0.5, tick)
t.start()
于 2017-01-03T19:05:27.990 に答える
4

swapnil-jariwalaコードのいくつかのコードを変更して、小さなコンソールクロックを作成しました。

from threading import Timer, Thread, Event
from datetime import datetime

class PT():

    def __init__(self, t, hFunction):
        self.t = t
        self.hFunction = hFunction
        self.thread = Timer(self.t, self.handle_function)

    def handle_function(self):
        self.hFunction()
        self.thread = Timer(self.t, self.handle_function)
        self.thread.start()

    def start(self):
        self.thread.start()

def printer():
    tempo = datetime.today()
    h,m,s = tempo.hour, tempo.minute, tempo.second
    print(f"{h}:{m}:{s}")


t = PT(1, printer)
t.start()

出力

>>> 11:39:11
11:39:12
11:39:13
11:39:14
11:39:15
11:39:16
...

tkinterグラフィックインターフェースを備えたタイマー

このコードは、tkinterを使用してクロックタイマーを小さなウィンドウに配置します

from threading import Timer, Thread, Event
from datetime import datetime
import tkinter as tk

app = tk.Tk()
lab = tk.Label(app, text="Timer will start in a sec")
lab.pack()


class perpetualTimer():

    def __init__(self, t, hFunction):
        self.t = t
        self.hFunction = hFunction
        self.thread = Timer(self.t, self.handle_function)

    def handle_function(self):
        self.hFunction()
        self.thread = Timer(self.t, self.handle_function)
        self.thread.start()

    def start(self):
        self.thread.start()

    def cancel(self):
        self.thread.cancel()


def printer():
    tempo = datetime.today()
    clock = "{}:{}:{}".format(tempo.hour, tempo.minute, tempo.second)
    try:
        lab['text'] = clock
    except RuntimeError:
        exit()


t = perpetualTimer(1, printer)
t.start()
app.mainloop()

フラッシュカードゲームの例(一種)

from threading import Timer, Thread, Event
from datetime import datetime


class perpetualTimer():

    def __init__(self, t, hFunction):
        self.t = t
        self.hFunction = hFunction
        self.thread = Timer(self.t, self.handle_function)

    def handle_function(self):
        self.hFunction()
        self.thread = Timer(self.t, self.handle_function)
        self.thread.start()

    def start(self):
        self.thread.start()

    def cancel(self):
        self.thread.cancel()


x = datetime.today()
start = x.second


def printer():
    global questions, counter, start
    x = datetime.today()
    tempo = x.second
    if tempo - 3 > start:
        show_ans()
    #print("\n{}:{}:{}".format(tempo.hour, tempo.minute, tempo.second), end="")
    print()
    print("-" + questions[counter])
    counter += 1
    if counter == len(answers):
        counter = 0


def show_ans():
    global answers, c2
    print("It is {}".format(answers[c2]))
    c2 += 1
    if c2 == len(answers):
        c2 = 0


questions = ["What is the capital of Italy?",
             "What is the capital of France?",
             "What is the capital of England?",
             "What is the capital of Spain?"]

answers = "Rome", "Paris", "London", "Madrid"

counter = 0
c2 = 0
print("Get ready to answer")
t = perpetualTimer(3, printer)
t.start()

出力:

Get ready to answer
>>> 
-What is the capital of Italy?
It is Rome

-What is the capital of France?
It is Paris

-What is the capital of England?
...
于 2017-08-09T09:40:17.880 に答える
2

私はプロジェクトのためにこれをしなければなりませんでした。私がやったことは、関数の別のスレッドを開始することでした

t = threading.Thread(target =heartbeat, args=(worker,))
t.start()

****ハートビートは私の機能であり、ワーカーは私の引数の1つです****

私の心拍機能の内部:

def heartbeat(worker):

    while True:
        time.sleep(5)
        #all of my code

したがって、スレッドを開始すると、関数は5秒間繰り返し待機し、すべてのコードを実行して、それを無期限に実行します。プロセスを強制終了する場合は、スレッドを強制終了します。

于 2017-03-25T18:53:53.693 に答える
1
from threading import Timer
def TaskManager():
    #do stuff
    t = Timer( 1, TaskManager )
    t.start()

TaskManager()

ここに小さなサンプルがあります、それはそれがどのように実行されるかを理解するのに役立ちます。最後に関数taskManager()は、それ自体への遅延関数呼び出しを作成します。

「dalay」変数を変更してみてください。違いがわかります。

from threading import Timer, _sleep

# ------------------------------------------
DATA = []
dalay = 0.25 # sec
counter = 0
allow_run = True
FIFO = True

def taskManager():

    global counter, DATA, delay, allow_run
    counter += 1

    if len(DATA) > 0:
        if FIFO:
            print("["+str(counter)+"] new data: ["+str(DATA.pop(0))+"]")
        else:
            print("["+str(counter)+"] new data: ["+str(DATA.pop())+"]")

    else:
        print("["+str(counter)+"] no data")

    if allow_run:
        #delayed method/function call to it self
        t = Timer( dalay, taskManager )
        t.start()

    else:
        print(" END task-manager: disabled")

# ------------------------------------------
def main():

    DATA.append("data from main(): 0")
    _sleep(2)
    DATA.append("data from main(): 1")
    _sleep(2)


# ------------------------------------------
print(" START task-manager:")
taskManager()

_sleep(2)
DATA.append("first data")

_sleep(2)
DATA.append("second data")

print(" START main():")
main()
print(" END main():")

_sleep(2)
DATA.append("last data")

allow_run = False
于 2016-06-20T18:52:35.940 に答える
1

タイマーとして機能するクラスを実装しました。

誰かがそれを必要とする場合に備えて、私はここにリンクを残します: https ://github.com/ivanhalencp/python/tree/master/xTimer

于 2017-06-28T17:15:03.823 に答える
1

私はright2clickyの答えが好きです。特に、スレッドを破棄する必要がなく、タイマーが作動するたびに新しいスレッドを作成する必要がないという点で気に入っています。さらに、定期的に呼び出されるタイマーコールバックを使用してクラスを作成するのは簡単なオーバーライドです。これが私の通常のユースケースです。

class MyClass(RepeatTimer):
    def __init__(self, period):
        super().__init__(period, self.on_timer)

    def on_timer(self):
        print("Tick")


if __name__ == "__main__":
    mc = MyClass(1)
    mc.start()
    time.sleep(5)
    mc.cancel()
于 2019-03-16T19:41:31.113 に答える
1

これは、クラスの代わりに関数を使用する代替実装です。上記の@AndrewWilkinsに触発されました。

待機はスリープよりも正確であるため(関数の実行時間を考慮に入れます):

import threading

PING_ON = threading.Event()

def ping():
  while not PING_ON.wait(1):
    print("my thread %s" % str(threading.current_thread().ident))

t = threading.Thread(target=ping)
t.start()

sleep(5)
PING_ON.set()
于 2019-07-02T14:51:42.050 に答える
1

SingleTonクラスを使用した別のソリューションを考え出しました。ここにメモリリークがあるかどうか教えてください。

import time,threading

class Singleton:
  __instance = None
  sleepTime = 1
  executeThread = False

  def __init__(self):
     if Singleton.__instance != None:
        raise Exception("This class is a singleton!")
     else:
        Singleton.__instance = self

  @staticmethod
  def getInstance():
     if Singleton.__instance == None:
        Singleton()
     return Singleton.__instance


  def startThread(self):
     self.executeThread = True
     self.threadNew = threading.Thread(target=self.foo_target)
     self.threadNew.start()
     print('doing other things...')


  def stopThread(self):
     print("Killing Thread ")
     self.executeThread = False
     self.threadNew.join()
     print(self.threadNew)


  def foo(self):
     print("Hello in " + str(self.sleepTime) + " seconds")


  def foo_target(self):
     while self.executeThread:
        self.foo()
        print(self.threadNew)
        time.sleep(self.sleepTime)

        if not self.executeThread:
           break


sClass = Singleton()
sClass.startThread()
time.sleep(5)
sClass.getInstance().stopThread()

sClass.getInstance().sleepTime = 2
sClass.startThread()
于 2020-07-25T05:06:59.370 に答える
1

スレッドを使用した上記の優れた回答に加えて、メインスレッドを使用する必要がある場合、または非同期アプローチを好む場合に備えて、aio_timersタイマークラスの周りに短いクラスをラップしました(繰り返しを有効にするため)

import asyncio
from aio_timers import Timer

class RepeatingAsyncTimer():
    def __init__(self, interval, cb, *args, **kwargs):
        self.interval = interval
        self.cb = cb
        self.args = args
        self.kwargs = kwargs
        self.aio_timer = None
        self.start_timer()
    
    def start_timer(self):
        self.aio_timer = Timer(delay=self.interval, 
                               callback=self.cb_wrapper, 
                               callback_args=self.args, 
                               callback_kwargs=self.kwargs
                              )
    
    def cb_wrapper(self, *args, **kwargs):
        self.cb(*args, **kwargs)
        self.start_timer()


from time import time
def cb(timer_name):
    print(timer_name, time())

print(f'clock starts at: {time()}')
timer_1 = RepeatingAsyncTimer(interval=5, cb=cb, timer_name='timer_1')
timer_2 = RepeatingAsyncTimer(interval=10, cb=cb, timer_name='timer_2')

時計の開始時刻:16024388 40 .9690785

timer_ 1 16024388 45 .980087

timer_ 2 16024388 50 .9806316

timer_ 1 16024388 50 .9808934

timer_ 1 16024388 55 .9863033

timer_ 2 16024388 60 .9868324

timer_ 1 16024388 60 .9876585

于 2020-10-11T17:59:31.163 に答える
0

これは、タイマーを継続的に実行するためのサンプルコードです。使い果たされたときに新しいタイマーを作成し、同じ関数を呼び出すだけです。それを行うための最良の方法ではありませんが、このようにすることもできます。

import threading
import time


class ContinousTimer():
    def __init__(self):
        self.timer = None

    def run(self, msg='abc'):
        print(msg)

        self.timer = threading.Timer(interval=2, function=self.run, args=(msg, ))
        self.timer.start()


if __name__ == "__main__":
    t = ContinousTimer()
    try:
        t.run(msg="Hello")
        while True:
            time.sleep(0.1)
    except KeyboardInterrupt:
        # Cancel Timer
        t.timer.cancel()
于 2021-09-22T10:34:16.797 に答える