Gibt es eine Möglichkeit, mehrere Teile des Programms zusammen laufen zu lassen, ohne mehrere Dinge im selben Codeblock zu tun?
Ein Thread wartet auf ein externes Gerät, während gleichzeitig eine LED in einem anderen Thread blinkt.
arduino-uno
threads
Bja
quelle
quelle
Antworten:
Auf dem Arduino gibt es keine Unterstützung für mehrere Prozesse und kein Multi-Threading. Mit einiger Software können Sie jedoch nahezu mehrere Threads ausführen.
Sie möchten sich Protothreads ansehen :
Natürlich gibt es ein Arduino Beispiel hier mit Beispielcode . Diese SO-Frage könnte auch nützlich sein.
ArduinoThread ist auch gut.
quelle
AVR-basierte Arduinos unterstützen kein (Hardware-) Threading. Ich bin mit ARM-basierten Arduinos nicht vertraut. Ein Weg, um diese Einschränkung zu umgehen, ist die Verwendung von Interrupts, insbesondere von zeitgesteuerten Interrupts. Sie können einen Timer programmieren, um die Hauptroutine alle so viele Mikrosekunden zu unterbrechen und eine bestimmte andere Routine auszuführen.
http://arduino.cc/en/Reference/Interrupts
quelle
Auf dem Uno ist softwareseitiges Multithreading möglich. Hardware Level Threading wird nicht unterstützt.
Um Multithreading zu erreichen, ist die Implementierung eines grundlegenden Schedulers und die Verwaltung eines Prozesses oder einer Aufgabenliste erforderlich, um die verschiedenen auszuführenden Aufgaben zu verfolgen.
Die Struktur eines sehr einfachen nicht präemptiven Schedulers sieht folgendermaßen aus:
Hierbei
tasklist
kann es sich um ein Array von Funktionszeigern handeln.Mit jeder Funktion des Formulars:
Jede Funktion kann eine separate Aufgabe
function1
ausführen, z. B. LED-Manipulationen undfunction2
Float-Berechnungen. Es liegt in der Verantwortung jeder Aufgabe (Funktion), die ihr zugewiesene Zeit einzuhalten.Hoffentlich sollte dies ausreichen, um Ihnen den Einstieg zu erleichtern.
quelle
Nach der Beschreibung Ihrer Anforderungen:
Es scheint, dass Sie einen Arduino-Interrupt für den ersten "Thread" verwenden könnten (ich würde es eher als "Task" bezeichnen).
Arduino-Interrupts können eine Funktion (Ihren Code) basierend auf einem externen Ereignis (Spannungspegel oder Pegeländerung an einem digitalen Eingangspin) aufrufen, die Ihre Funktion sofort auslöst.
Ein wichtiger Punkt bei Interrupts ist jedoch, dass die aufgerufene Funktion so schnell wie möglich ist (normalerweise sollte es keinen
delay()
Aufruf oder eine andere API geben, die davon abhängtdelay()
).Wenn Sie eine lange Aufgabe haben, die beim Auslösen eines externen Ereignisses aktiviert werden muss, können Sie möglicherweise einen kooperativen Scheduler verwenden und dieser von Ihrer Interrupt-Funktion aus eine neue Aufgabe hinzufügen.
Ein zweiter wichtiger Punkt bei Interrupts ist, dass ihre Anzahl begrenzt ist (z. B. nur 2 bei UNO). Wenn Sie also mehr externe Ereignisse haben möchten, müssen Sie eine Art Multiplexing aller Eingänge in einen implementieren und Ihre Interrupt-Funktion bestimmen lassen, welcher Multiplexing-Eingang der eigentliche Auslöser war.
quelle
Eine einfache Lösung ist die Verwendung eines Schedulers . Es gibt mehrere Implementierungen. Dies beschreibt in Kürze eine, die für AVR- und SAM-basierte Karten verfügbar ist. Grundsätzlich startet ein einzelner Anruf eine Aufgabe. msgstr "innerhalb einer Skizze skizzieren".
Scheduler.start () fügt eine neue Aufgabe hinzu, die das taskSetup einmal ausführt und dann wiederholt taskLoop aufruft, genau wie die Arduino-Skizze funktioniert. Die Aufgabe hat einen eigenen Stapel. Die Größe des Stapels ist ein optionaler Parameter. Die Standardstapelgröße beträgt 128 Byte.
Um Kontextwechsel zu ermöglichen, müssen die Tasks yield () oder delay () aufrufen . Es gibt auch ein Unterstützungsmakro für das Warten auf eine Bedingung.
Das Makro ist syntaktischer Zucker für Folgendes:
Warten kann auch zum Synchronisieren von Aufgaben verwendet werden. Unten ist ein Beispiel-Snippet:
Weitere Details finden Sie in den Beispielen . Es gibt Beispiele für das Blinken mehrerer LEDs, um die Schaltfläche zu entprellen, und eine einfache Shell mit nicht blockierendem Befehlszeilenlesevorgang. Vorlagen und Namespaces können zum Strukturieren und Reduzieren des Quellcodes verwendet werden. Die folgende Skizze zeigt, wie Vorlagenfunktionen für Multi-Blink verwendet werden. Es reicht mit 64 Bytes für den Stack.
Es gibt auch einen Benchmark , der eine Vorstellung von der Leistung gibt, dh Zeit zum Starten der Aufgabe, Kontextwechsel usw.
Zuletzt gibt es einige Support-Klassen für die Synchronisierung und Kommunikation auf Task-Ebene. Warteschlange und Semaphor .
quelle
Aus einer früheren Beschwörung dieses Forums wurde die folgende Frage / Antwort nach Elektrotechnik verschoben. Es hat Beispiel-Arduino-Code, um eine LED mit einem Timer-Interrupt zu blinken, während die Hauptschleife für serielle E / A verwendet wird.
https://electronics.stackexchange.com/questions/67089/how-can-i-control-things-without-using-delay/67091#67091
Repost:
Interrupts sind eine übliche Methode, um Dinge zu erledigen, während etwas anderes läuft. Im folgenden Beispiel blinkt die LED ohne Verwendung von
delay()
. Bei jedemTimer1
Brand wird die Interrupt Service Routine (ISR)isrBlinker()
aufgerufen. Es schaltet die LED ein / aus.Um zu zeigen, dass andere Dinge gleichzeitig passieren können, wird
loop()
wiederholt foo / bar auf die serielle Schnittstelle geschrieben, unabhängig davon , ob die LED blinkt.Dies ist eine sehr einfache Demo. ISRs können sehr viel komplexer sein und durch Timer und externe Ereignisse (Pins) ausgelöst werden. Viele der gängigen Bibliotheken werden mithilfe von ISRs implementiert.
quelle
Ich bin auch auf dieses Thema gekommen, als ich eine Matrix-LED-Anzeige implementiert habe.
Mit einem Wort, Sie können einen Abrufplaner erstellen, indem Sie in Arduino die Funktion millis () und den Timer-Interrupt verwenden.
Ich schlage folgende Artikel von Bill Earl vor:
https://learn.adafruit.com/multi-tasking-the-arduino-part-1/overview
https://learn.adafruit.com/multi-tasking-the-arduino-part-2/overview
https://learn.adafruit.com/multi-tasking-the-arduino-part-3/overview
quelle
Sie könnten auch meine ThreadHandler-Bibliothek ausprobieren
https://bitbucket.org/adamb3_14/threadhandler/src/master/
Es verwendet einen unterbrechenden Scheduler, um das Umschalten des Kontexts zu ermöglichen, ohne auf yield () oder delay () zu verweisen.
Ich habe die Bibliothek erstellt, weil ich drei Threads brauchte und zwei von ihnen, um genau zu laufen, egal was die anderen taten. Der erste Thread behandelte die serielle Kommunikation. Im zweiten Beispiel wurde ein Kalman-Filter mit Float-Matrix-Multiplikation mit der Eigen-Bibliothek ausgeführt. Und der dritte war ein schneller Stromregelkreis-Thread, der in der Lage sein musste, die Matrixberechnungen zu unterbrechen.
Wie es funktioniert
Jeder zyklische Thread hat eine Priorität und eine Periode. Wenn ein Thread mit höherer Priorität als der aktuell ausgeführte Thread seine nächste Ausführungszeit erreicht, hält der Scheduler den aktuellen Thread an und wechselt zu dem Thread mit der höheren Priorität. Sobald der Thread mit hoher Priorität seine Ausführung abgeschlossen hat, wechselt der Scheduler zurück zum vorherigen Thread.
Planungsregeln
Das Planungsschema der ThreadHandler-Bibliothek lautet wie folgt:
Wie benutzt man
Threads können über C ++ - Vererbung erstellt werden
Oder über createThread und eine Lambda-Funktion
Thread-Objekte stellen beim Erstellen automatisch eine Verbindung zum ThreadHandler her.
So starten Sie die Ausführung von erstellten Thread-Objekten:
quelle
Und hier ist noch eine weitere kooperative Mikroprozessor-Multitasking-Bibliothek - PQRST: eine Prioritätswarteschlange zum Ausführen einfacher Aufgaben.
In diesem Modell wird ein Thread als Unterklasse von a implementiert
Task
, die für einen späteren Zeitpunkt geplant ist (und möglicherweise in regelmäßigen Abständen neu geplant wird, wenn es wie üblichLoopTask
stattdessen Unterklassen gibt ). Dierun()
Methode des Objekts wird aufgerufen, wenn die Aufgabe fällig wird. Dierun()
Methode erledigt einige fällige Arbeiten und gibt dann zurück (dies ist das kooperative Bit). In der Regel wird eine Art Zustandsmaschine verwaltet, um die Aktionen bei aufeinanderfolgenden Aufrufen zu verwalten (ein einfaches Beispiel ist dielight_on_p_
Variable im folgenden Beispiel). Es erfordert ein leichtes Überdenken der Code-Organisation, hat sich jedoch bei intensiver Nutzung als sehr flexibel und robust erwiesen.Es ist agnostisch in Bezug auf die Zeiteinheiten, so dass es genauso glücklich ist, in Einheiten von
millis()
wiemicros()
oder jedem anderen Tick zu laufen, der praktisch ist.Hier ist das 'Blink'-Programm, das mit dieser Bibliothek implementiert wurde. Dies zeigt nur eine einzige ausgeführte Aufgabe: In der Regel werden andere Aufgaben erstellt und innerhalb von gestartet
setup()
.quelle
run()
Methode aufgerufen wurde, wird sie nicht unterbrochen, sodass sie die Verantwortung hat, angemessen schnell fertig zu werden. In der Regel erledigt es jedoch seine Arbeit und plant sich dannLoopTask
für einige Zeit selbst neu (möglicherweise automatisch, im Fall einer Unterklasse von ). Ein gängiges Muster besteht darin, dass die Aufgabe eine interne Zustandsmaschine verwaltet (ein einfaches Beispiel ist derlight_on_p_
obige Zustand), damit sie sich bei nächster Fälligkeit angemessen verhält.run()
. Dies steht im Gegensatz zu kooperativen Threads, die die CPU beispielsweise durch Aufrufen vonyield()
oder ausgeben könnendelay()
. Oder präventive Threads, die jederzeit terminiert werden können. Ich halte die Unterscheidung für wichtig, da ich gesehen habe, dass viele Leute, die hier nach Threads suchen, dies tun, weil sie es vorziehen, blockierenden Code zu schreiben, anstatt Zustandsautomaten. Das Blockieren von echten Threads, die die CPU belasten, ist in Ordnung. Das Blockieren von RtC-Aufgaben ist nicht möglich.