<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="de">
	<id>https://wiki-de.moshellshocker.dns64.de/index.php?action=history&amp;feed=atom&amp;title=Semaphor_%28Informatik%29</id>
	<title>Semaphor (Informatik) - Versionsgeschichte</title>
	<link rel="self" type="application/atom+xml" href="https://wiki-de.moshellshocker.dns64.de/index.php?action=history&amp;feed=atom&amp;title=Semaphor_%28Informatik%29"/>
	<link rel="alternate" type="text/html" href="https://wiki-de.moshellshocker.dns64.de/index.php?title=Semaphor_(Informatik)&amp;action=history"/>
	<updated>2026-06-03T06:51:09Z</updated>
	<subtitle>Versionsgeschichte dieser Seite in Wikipedia (Deutsch) – Lokale Kopie</subtitle>
	<generator>MediaWiki 1.43.8</generator>
	<entry>
		<id>https://wiki-de.moshellshocker.dns64.de/index.php?title=Semaphor_(Informatik)&amp;diff=60139&amp;oldid=prev</id>
		<title>~2026-79915-4 am 5. Februar 2026 um 13:42 Uhr</title>
		<link rel="alternate" type="text/html" href="https://wiki-de.moshellshocker.dns64.de/index.php?title=Semaphor_(Informatik)&amp;diff=60139&amp;oldid=prev"/>
		<updated>2026-02-05T13:42:11Z</updated>

		<summary type="html">&lt;p&gt;&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Neue Seite&lt;/b&gt;&lt;/p&gt;&lt;div&gt;Ein &amp;#039;&amp;#039;&amp;#039;Semaphor&amp;#039;&amp;#039;&amp;#039; (von {{grcS|σῆμα|&amp;#039;&amp;#039;sēma&amp;#039;&amp;#039;|de=Zeichen}} und {{lang|grc|φέρειν}} &amp;#039;&amp;#039;pherein&amp;#039;&amp;#039; ‚tragen‘ – also etwa „Signalgeber“) ist eine [[Datenstruktur]], die aus einer Ganzzahl und den atomaren Nutzungsoperationen „Reservieren/Probieren“ und „Freigeben“ besteht. Sie eignet sich insbesondere zur Verwaltung beschränkter (zählbarer) Ressourcen, auf die mehrere [[Prozess (Informatik)|Prozesse]] oder [[Thread (Informatik)|Threads]] zugreifen sollen, wie etwa Erzeuger und Verbraucher, sowie zur Koordination asynchroner Abläufe. Im Gegensatz zu einem [[Lock]] bzw. einem [[Mutex]] müssen die Aktivitätsträger, die „reservieren“ und „freigeben“, nicht identisch sein.&lt;br /&gt;
&lt;br /&gt;
== Funktionsweise ==&lt;br /&gt;
Meist wird die Ganzzahl (Zähler) beim Start des Semaphors mit dem Zahlenwert der maximal verfügbaren Ressourcen initialisiert bzw. der maximalen Zahl der Prozesse, die gleichzeitig die Ressource nutzen können. Ein Prozess, der auf die Ressource zugreifen will, muss vorher die Operation „Reservieren/Probieren“ aufrufen, und danach, wenn er die Ressource nicht mehr benötigt, die Operation „Freigeben“. Bei jeder Reservierung wird der Zähler um 1 heruntergezählt, bei Freigabe wird er wieder um 1 erhöht. Der Zähler darf nicht unter 0 fallen: Wenn eine Reservierung bei Zählerstand 0 erfolgt, wartet der reservierende Prozess, bis ein anderer Prozess Ressourcen freigegeben hat. Es gibt auch Implementierungen, die ins Negative zählen. Damit kann angezeigt werden, wie viele Prozesse auf eine Freigabe warten.&lt;br /&gt;
&lt;br /&gt;
Semaphore können bei der [[Programmierung]] zur [[Prozesssynchronisation]] eingesetzt werden, also zur Lösung von Aufgaben, bei denen die parallele Ausführung mehrerer [[Prozess (Informatik)|Prozesse]]/[[Thread (Informatik)|Threads]] eine zeitliche Abstimmung der Ausführungen erfordert. Sie dienen im Allgemeinen dazu, bei einer beschränkten Anzahl von Ressourcen eine Reihenfolge herzustellen, in der viele Threads sich diese knappen Elemente teilen (z.&amp;amp;nbsp;B. Ressource: „CPU-Kern“, Anzahl: 4). Dies kann auch zur Kapselung von Zugriffen auf gemeinsame Daten verwendet werden (Ressource: „Zugriffsrecht auf die Daten“, Anzahl: immer nur einer gleichzeitig). Auch zur Kommunikation zwischen Threads können Semaphore verwendet werden. Sie dienen dann meist als Zähler für verfügbare Informationspakete. Hierbei wird der Semaphor mit „0 Pakete verfügbar“ gestartet, und dann hochgezählt (und wieder bis auf 0 herunter).&lt;br /&gt;
&lt;br /&gt;
== Namensherkunft ==&lt;br /&gt;
[[Datei:Velgast-suedbahn.jpg|mini|hochkant|Verlassenes Eisenbahnsignal (Semaphor)]]&lt;br /&gt;
Das Wort &amp;#039;&amp;#039;Semaphor&amp;#039;&amp;#039; geht auf die [[Formsignal]]e mechanischer [[Eisenbahnsignal]]e zurück.&amp;lt;ref&amp;gt;[https://tldp.org/LDP/lpg/node47.html &amp;#039;&amp;#039;Semaphores: Basic Concepts&amp;#039;&amp;#039;.] In: &amp;#039;&amp;#039;The Linux Programmer’s Guide&amp;#039;&amp;#039;&amp;lt;/ref&amp;gt; Die dort verwendeten &amp;#039;&amp;#039;Semaphore&amp;#039;&amp;#039; zeigen an, ob ein [[Zug (Schienenverkehr)|Zug]] eine Ressource belegt, also ob ein [[Zugfolgeabschnitt|Gleisabschnitt]] befahren werden darf oder nicht.&lt;br /&gt;
&lt;br /&gt;
== Wechselwirkungen parallel ablaufender Prozesse ==&lt;br /&gt;
Bei der parallelen oder zeitlich verzahnten Ausführung von Prozessen treten implizite oder explizite Wechselwirkungen auf.&lt;br /&gt;
&lt;br /&gt;
Bei impliziten Wechselwirkungen ist einem Prozess nicht bewusst, dass durch die Ausführung von Aktionen ein anderer Prozess beeinflusst wird. Dies ist z.&amp;amp;nbsp;B. dann der Fall, wenn ein Prozess einen Systemdienst aufruft, den das Betriebssystem nicht sofort vollständig bearbeiten kann, weil andere Prozesse die erforderlichen Betriebsmittel belegt haben. Der Prozess kann erst dann seine Aktionen weiter fortsetzen, wenn der Systemdienst ausgeführt worden ist. Hier wird die Prozesswechselwirkung als blockierender Funktionsaufruf sichtbar. Spezielle Vorkehrungen gegen eine Blockierung aufgrund impliziter Wechselwirkungen muss und kann ein Prozess nicht treffen.&lt;br /&gt;
&lt;br /&gt;
Explizite Wechselwirkungen zwischen Prozessen sind:&lt;br /&gt;
; Konkurrenz&lt;br /&gt;
: Prozesse stehen in Konkurrenz zueinander, wenn sie gleichzeitig auf ein Betriebsmittel (z.&amp;amp;nbsp;B. Speicherstruktur, Verbindung, Gerät) zugreifen, das nur in beschränkter Anzahl zur Verfügung steht und bei dem die Nutzung eines Exemplars nur exklusiv durch einen Prozess möglich ist, da es andernfalls zu fehlerhaften Ergebnissen oder inkonsistenten Zuständen kommt, d.&amp;amp;nbsp;h. wenn es [[Kritischer Abschnitt|kritische Abschnitte]] in den Programmen der Prozesse gibt.&lt;br /&gt;
; Kooperation&lt;br /&gt;
: Prozesse kooperieren, wenn sie ihre Aktionen bewusst aufeinander abstimmen, z.&amp;amp;nbsp;B. weil sie in einer Auftraggeber-/Auftragnehmerbeziehung stehen.&lt;br /&gt;
&lt;br /&gt;
Sowohl das Reservieren als auch das Freigeben müssen [[atomar]]e Operationen sein. Kann eine Reservierung nicht befriedigt werden, so kann sie einfach blockieren (Erlangung der Ressource via [[Race Condition]] unter den Wartenden), der Semaphor eine Warteschlange führen (i. A. blockierend) oder ablehnen (nicht blockierend).&lt;br /&gt;
Häufig ist vom Betriebsmittel nur &amp;#039;&amp;#039;ein&amp;#039;&amp;#039; Exemplar vorhanden (sog. [[Mutex|wechselseitiger Ausschluss]]), der Semaphor bewirkt dann eine Abstimmung der zeitlichen Ausführung der Prozessaktionen. Im Fall einer Konkurrenzsituation wird durch eine irgendwie gestaltete Sequentialisierung der Ausführung (der kritischen Abschnitte) erreicht, dass das Betriebsmittel nicht von mehreren Prozessen beliebig verändernd benutzt wird. Im Fall einer Kooperationssituation wird ebenfalls durch eine der Situation entsprechende Sequentialisierung erreicht, dass die Zusammenarbeit der Prozesse gegeben ist (z.&amp;amp;nbsp;B., dass ein Auftragnehmer nicht schon versucht anzufangen etwas zu bearbeiten, obwohl der Auftraggeber noch keinen Auftrag erteilt hat).&lt;br /&gt;
&lt;br /&gt;
== Lösung von Dijkstra ==&lt;br /&gt;
Semaphore als Mechanismus für die Prozesssynchronisation wurden von [[Edsger W. Dijkstra]] konzipiert und 1965 in seinem Artikel &amp;#039;&amp;#039;Cooperating sequential processes&amp;#039;&amp;#039; vorgestellt.&lt;br /&gt;
&lt;br /&gt;
Ein Semaphor ist eine Datenstruktur mit einer Initialisierungsoperation und zwei Nutzungsoperationen. Die Datenstruktur besteht aus einem Zähler und einer [[Warteschlange (Datenstruktur)|Warteschlange]] für die Aufnahme blockierter Prozesse (hier ein Beispiel in [[C (Programmiersprache)|C]]):&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;c&amp;quot;&amp;gt;&lt;br /&gt;
 typedef struct semaphor {&lt;br /&gt;
    int zaehler;&lt;br /&gt;
    Queue *queue; /* Warteschlange */&lt;br /&gt;
 } Semaphor;&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Zähler sowie Warteschlange sind geschützt und können nur über die Semaphoroperationen verändert werden. Die Wirkung der Nutzungsoperation kann wie folgt zusammenfassend beschrieben werden:&lt;br /&gt;
* Semaphore regeln Wechselwirkungssituationen von Prozessen durch Zählen&lt;br /&gt;
* Semaphore realisieren ein passives Warten der Prozesse, wenn eine Weiterausführung nicht gestattet werden kann&lt;br /&gt;
&lt;br /&gt;
Mit der Initialisierungsoperation wird der Zähler auf einen nicht negativen Wert (≥&amp;amp;nbsp;0) und die Warteschlange i.&amp;amp;nbsp;d.&amp;amp;nbsp;R. auf leer gesetzt.&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;c&amp;quot;&amp;gt;&lt;br /&gt;
 void&lt;br /&gt;
 init(Semaphor  *s, int v)&lt;br /&gt;
 {&lt;br /&gt;
    s-&amp;gt;zaehler = v;&lt;br /&gt;
    s-&amp;gt;queue-&amp;gt;empty(s-&amp;gt;queue);&lt;br /&gt;
 }&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Wird ein Semaphor zur Organisation von Konkurrenzsituationen eingesetzt, so erfolgt eine Initialisierung mit einem positiven Wert. Ein Semaphor für eine Kooperationssituation wird hingegen mit 0 initialisiert (siehe [[#Anwendungsbeispiele|Anwendungsbeispiele]]).&lt;br /&gt;
&lt;br /&gt;
Die Nutzungsoperationen wurden von Dijkstra mit &amp;#039;&amp;#039;P&amp;#039;&amp;#039; und &amp;#039;&amp;#039;V&amp;#039;&amp;#039; bezeichnet. Dies sind Initialen [[Niederländische Sprache|niederländischer]] Wörter bzw. [[Kofferwort|Kofferwörter]] für &amp;#039;&amp;#039;prolaag&amp;#039;&amp;#039; (&amp;#039;&amp;#039;probeer te verlagen&amp;#039;&amp;#039; = versuche zu senken) und &amp;#039;&amp;#039;verhoog&amp;#039;&amp;#039;.&amp;lt;ref&amp;gt;[https://www.cs.utexas.edu/users/EWD/transcriptions/EWD00xx/EWD74.html cs.utexas.edu] E. W. Dijkstra Archive (niederländisch)&amp;lt;/ref&amp;gt; Weitere, verbreitete Erklärungen sind &amp;#039;&amp;#039;passeer&amp;#039;&amp;#039; (passieren), &amp;#039;&amp;#039;proberen&amp;#039;&amp;#039; (überprüfen)&amp;lt;ref name=&amp;quot;Silberschatz_S200&amp;quot;&amp;gt;{{Literatur |Autor=Abraham Silberschatz, Peter B. Galvin, Greg Gagne |Titel=Operating system concepts |Auflage=7 |Verlag=John Wiley &amp;amp; Sons |Ort=Hoboken NJ |Datum=2005 |ISBN=0-471-69466-5 |Kapitel=6.5 Semaphores |Seiten=200 |Sprache=en}}&amp;lt;/ref&amp;gt; und &amp;#039;&amp;#039;vrijgeven&amp;#039;&amp;#039; (freigeben), &amp;#039;&amp;#039;verhogen&amp;#039;&amp;#039; (erhöhen).&amp;lt;ref name=&amp;quot;Silberschatz_S200&amp;quot; /&amp;gt; Programmierschnittstellen verwenden mnemonisch deutlichere Bezeichnungen wie &amp;#039;&amp;#039;wait&amp;#039;&amp;#039; (warten), &amp;#039;&amp;#039;acquire&amp;#039;&amp;#039; (erlangen) oder &amp;#039;&amp;#039;down&amp;#039;&amp;#039; (unten) für die &amp;#039;&amp;#039;P&amp;#039;&amp;#039;-Operation und &amp;#039;&amp;#039;signal&amp;#039;&amp;#039; (signalisieren), &amp;#039;&amp;#039;release&amp;#039;&amp;#039; (freigeben), &amp;#039;&amp;#039;post&amp;#039;&amp;#039; (abschicken)&amp;lt;ref&amp;gt;[https://pubs.opengroup.org/onlinepubs/7908799/xsh/sem_post.html pubs.opengroup.org]&amp;lt;/ref&amp;gt; oder &amp;#039;&amp;#039;up&amp;#039;&amp;#039; (oben) für die &amp;#039;&amp;#039;V&amp;#039;&amp;#039;-Operation.&lt;br /&gt;
&lt;br /&gt;
Bei einem Aufruf der &amp;#039;&amp;#039;P&amp;#039;&amp;#039;-Operation wird der Zähler dekrementiert. Ist der Zähler danach größer gleich 0, so setzt der Prozess seine Aktionen fort. Ist der Zähler jedoch kleiner als 0, kehrt der Kontrollfluss nicht aus der Operation zurück. Der aufrufende Prozess wird blockiert und in die Warteschlange des Semaphors eingereiht. Bei einem Aufruf der &amp;#039;&amp;#039;V&amp;#039;&amp;#039;-Operation wird der Zähler inkrementiert. Es wird ein Prozess aus der Warteschlange entnommen und entblockiert, falls die Warteschlange nicht leer ist. Der entblockierte Prozess setzt dann seine Aktionen mit denen fort, die dem &amp;#039;&amp;#039;P&amp;#039;&amp;#039;-Aufruf folgen, der den Prozess blockierte. Die hier erläuterte Funktionsweise der Semaphoroperationen erlaubt eine einfache Ermittlung, ob es Prozesse gibt, die am Semaphor blockiert sind: ist der Semaphorzähler kleiner als 0, so gibt es noch blockierte Prozesse. Diese Prozesse riefen die P-Operation auf, als der Zähler bereits kleiner gleich 0 war. Die Überprüfung wird hier zwecks Verdeutlichung der Wirkung der V-Operation auf andere Prozesse explizit notiert. Konkrete Implementierungen können eine Prüfung auf nicht-leere Warteschlange auch in die Warteschlangenmethode verlagern.&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;c&amp;quot;&amp;gt;&lt;br /&gt;
/* down() */&lt;br /&gt;
 void&lt;br /&gt;
 P(Semaphor *s)&lt;br /&gt;
 {&lt;br /&gt;
   s-&amp;gt;zaehler = s-&amp;gt;zaehler - 1;&lt;br /&gt;
   if (s-&amp;gt;zaehler &amp;lt; 0)&lt;br /&gt;
      selbst_blockieren(s-&amp;gt;queue); /* Blockieren des Prozesses, Einreihung in Warteschlange */&lt;br /&gt;
 }&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;c&amp;quot;&amp;gt;&lt;br /&gt;
/* up() */&lt;br /&gt;
 void&lt;br /&gt;
 V(Semaphor *s)&lt;br /&gt;
 {&lt;br /&gt;
   if (s-&amp;gt;zaehler &amp;lt;= 0)&lt;br /&gt;
      einen_entblocken(s-&amp;gt;queue); /* Entblockieren eines Prozesses aus der Warteschlange */&lt;br /&gt;
   s-&amp;gt;zaehler = s-&amp;gt;zaehler + 1;&lt;br /&gt;
 }&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Semaphore, deren Zähler aufgrund der Initialisierung und der Verwendung eine &amp;quot;1&amp;quot; als größten positiven Wert annehmen können, werden oftmals auch als binäre Semaphore bzw. Mutex locks bezeichnet.&amp;lt;ref name=&amp;quot;Silberschatz_S201&amp;quot;&amp;gt;{{Literatur |Autor=Abraham Silberschatz, Peter B. Galvin, Greg Gagne |Titel=Operating system concepts |Auflage=7 |Verlag=John Wiley &amp;amp; Sons |Ort=Hoboken NJ |Datum=2005 |ISBN=0-471-69466-5 |Kapitel=6.5 Semaphores |Seiten=201 |Sprache=en}}&amp;lt;/ref&amp;gt; Semaphore, deren Zähler größere positive Werte als &amp;quot;1&amp;quot; annehmen können, werden zählende Semaphore genannt.&lt;br /&gt;
&lt;br /&gt;
Beide Operationen müssen unteilbare, [[Atomar#Informatik|atomare]] Aktionen sein. Dadurch ist garantiert, dass nach dem Aufruf einer Operation eines Semaphors kein anderer Prozess auf den gleichen Semaphor durch einen Operationsaufruf modifizierend zugreifen kann, bevor die zuerst aufgerufene Semaphoroperation vollständig ausgeführt worden ist. Die Unteilbarkeit ist notwendig, um die Synchronisation zu organisieren und [[Race Condition|Wettlaufsituationen]] bei Ausführung der Semaphoroperationen durch parallele Prozesse zu vermeiden.&lt;br /&gt;
&lt;br /&gt;
Die obige Erläuterung der Arbeitsweise ist eine von mehreren möglichen. Diese unterscheiden sich durch die Reihenfolge der Prüfung auf Blockierung/Entblockierung und der Operation Inkrement/Dekrement. In einigen Darstellungen nimmt der Zähler keine negativen Werte an. (Der oben beschriebene Zählerwert ergibt sich, indem man vom Tatsächlichen die Länge der Warteschlange subtrahiert.) In diesem Fall wird die Bezeichnung binärer Semaphor dann offensichtlich.&lt;br /&gt;
Die Wirkung der Operationen auf Prozesse ist jedoch unabhängig von der Art der Realisierung. Der obigen Erläuterung wurde wegen einer einfachen Interpretation des Zählers der Vorzug gegeben: Ist der Zähler größer als 0, so gibt sein Wert an, wie viele Prozesse noch ohne Blockierung die P-Operation aufrufen können. Ist der Zähler negativ, so gibt sein Absolutwert an, wie viele Prozesse die P-Operation aufgerufen haben und dabei blockiert wurden.&lt;br /&gt;
&lt;br /&gt;
Semaphore beheben den Nachteil des [[Aktives Warten|aktiven Wartens]] anderer Synchronisationslösungen wie spezielle Maschinenbefehle oder [[Spinlock]]s, da ein Prozess blockiert wird, bis der Blockadegrund entfallen ist. Die Definition lässt offen, welcher blockierte Prozess im Rahmen der Ausführung der &amp;#039;&amp;#039;V&amp;#039;&amp;#039;-Operation der Warteschlange entnommen wird. Ein Semaphor, der eine echte Warteschlange nach dem [[Windhundprinzip]] (&amp;#039;&amp;#039;engl. „first come, first served“&amp;#039;&amp;#039;) garantiert, wird manchmal als starker Semaphor bezeichnet. Ein schwacher Semaphor garantiert hingegen nicht die chronologisch richtige Abarbeitung der Warteschlange. So wird z.&amp;amp;nbsp;B. beim Echtzeitbetrieb das Entblockieren von Prozessen an deren Priorität statt an deren Blockierungszeitpunkt gebunden.&lt;br /&gt;
&lt;br /&gt;
== Anwendungsbeispiele ==&lt;br /&gt;
===Einsatz in Konkurrenzsituationen===&lt;br /&gt;
In einem [[Kritischer Abschnitt|kritischen Abschnitt]] &amp;#039;&amp;#039;ka_A&amp;#039;&amp;#039; verändert Prozess A eine [[Datenstruktur]], die der [[Prozess (Informatik)|Prozess]] gemeinsam mit einem Prozess B nutzt. Prozess B verändert die Datenstruktur in seinem kritischen Abschnitt &amp;#039;&amp;#039;ka_B&amp;#039;&amp;#039;. Ein [[Mutex|Semaphor in Ausschlussfunktion]] wird eingesetzt, um zu erreichen, dass sich die Prozesse A und B niemals gleichzeitig in ihren kritischen Abschnitten befinden. Hierzu wird der Semaphor mit 1 initialisiert, es wird also ein binärer Semaphor eingesetzt:&lt;br /&gt;
 Gemeinsam von A und B genutzter Semaphor: mutex&lt;br /&gt;
 Initialisierung: init (mutex, 1)  /* Es gibt 1 Ressourcen-Exemplar &amp;quot;Recht auf Betreten eines kritischen Abschnitts&amp;quot; */&lt;br /&gt;
&lt;br /&gt;
 Prozess A           Prozess B&lt;br /&gt;
 ...                 ...&lt;br /&gt;
 P (mutex) P (mutex) /* versuche &amp;quot;Betreten-Recht&amp;quot; zu reservieren (blockierend) */&lt;br /&gt;
 ka_A                ka_B&lt;br /&gt;
 V (mutex) V (mutex) /* freigeben des &amp;quot;Betreten-Rechts&amp;quot; */&lt;br /&gt;
 ...                 ...&lt;br /&gt;
Benötigen [[Prozess (Informatik)|Prozesse]] jeweils exklusiv ein [[Betriebsmittel (Informatik)|Betriebsmittel]], das nur in beschränkter Anzahl zur Verfügung steht, so wird mittels eines zählenden Semaphors erreicht, dass ein Prozess dann blockiert wird, wenn alle Betriebsmittel in Benutzung sind. Der Semaphor wird mit der Anzahl verfügbarer Betriebsmittel initialisiert:&lt;br /&gt;
 Semaphor zur Betriebsmittelverwaltung: s_available&lt;br /&gt;
 Initialisierung: init (s_available, n)&lt;br /&gt;
&lt;br /&gt;
 Prozess&lt;br /&gt;
 ...&lt;br /&gt;
 P (s_available) /* Anzeige des Nutzungswunschs; warte bis &amp;quot;für mich reserviert wurde&amp;quot; */&lt;br /&gt;
 ... /* Nutzung des Betriebsmittels */&lt;br /&gt;
 V (s_available) /* Anzeige des Nutzungsabschlusses, freigeben */&lt;br /&gt;
 ...&lt;br /&gt;
&lt;br /&gt;
===Einsatz in Kooperationssituationen===&lt;br /&gt;
[[Prozess (Informatik)|Prozess]] A enthält einen Programmabschnitt &amp;#039;&amp;#039;C_I&amp;#039;&amp;#039;, in dem eine Datenstruktur initialisiert wird, die dann von Prozess B in einem Programmabschnitt &amp;#039;&amp;#039;C_V&amp;#039;&amp;#039; verarbeitet wird. Offensichtlich muss &amp;#039;&amp;#039;C_I&amp;#039;&amp;#039; unter allen Umständen vor &amp;#039;&amp;#039;C_V&amp;#039;&amp;#039; ausgeführt werden. Es wird ein Semaphor in Signalisierungsfunktion eingesetzt:&lt;br /&gt;
 Gemeinsam von A und B genutzter Semaphor: s_inform&lt;br /&gt;
 Initialisierung: init (s_inform, 0)&lt;br /&gt;
&lt;br /&gt;
 Prozess A           Prozess B&lt;br /&gt;
 ...                 ...&lt;br /&gt;
 C_I                 P (s_inform)&lt;br /&gt;
 V (s_inform)        C_V&lt;br /&gt;
 ...                 ...&lt;br /&gt;
&lt;br /&gt;
Prozess B versucht zu reservieren und muss warten, bis Prozess A mittels Freigabe &amp;#039;&amp;#039;s_inform&amp;#039;&amp;#039; auf 1 (Datensatz verfügbar) hochgezählt hat.&lt;br /&gt;
&lt;br /&gt;
===Passing the Baton Pattern===&lt;br /&gt;
&lt;br /&gt;
Das von Gregory R. Andrews vorgeschlagene &amp;#039;&amp;#039;Passing the Baton Pattern&amp;#039;&amp;#039;&amp;lt;ref&amp;gt;{{cite book |last1=Andrews |first1=Gregory R. | date=1999|language=en|publisher=Addison-Wesley|title=Foundations of Multithreaded, Parallel, and Distributed Programming}}&amp;lt;/ref&amp;gt;&amp;lt;ref&amp;gt;{{cite book |last1=Carver |first1=Richard H. |last2=Thai |first2=Kuo-Chung |date=2005|language=en|publisher=Wiley|title=Modern Multithreading: Implementing, Testing, and Debugging Multithreaded Java and C++/Pthreads/Win32 Programs}}&amp;lt;/ref&amp;gt;&amp;lt;ref&amp;gt;https://greenteapress.com/wp/semaphores/&amp;lt;/ref&amp;gt; (deutsch: &amp;#039;&amp;#039;Staffelstab-Algorithmus&amp;#039;&amp;#039;&amp;lt;ref&amp;gt;{{cite book |last1=Maurer |first1=Christian | date=2021|language=en|publisher=Springer|title=Nonsequential and Distributed Programming with Go}}&amp;lt;/ref&amp;gt;) ist ein [[Entwurfsmuster|allgemeines Schema]] zur Lösung vieler [[Generische Programmierung|generischer]] Programmierprobleme, bei denen mehrere [[Nebenläufigkeit|nebenläufige]] [[Prozess (Informatik)|Prozesse]] um dieselbe Ressource mit komplexen Bedingungssynchronisationen konkurrieren, z.&amp;amp;nbsp;B. Erfüllung bestimmter Prioritätskriterien oder Vermeidung von [[Verhungern (Informatik)|Verhungern]]. Bei einer gemeinsam genutzten Ressource erfordert das Pattern eine private Semaphore &amp;#039;&amp;#039;priv&amp;#039;&amp;#039; (initialisiert auf 0) für jeden beteiligten Prozess oder jede Klasse von Prozessen und eine einzige Semaphore &amp;#039;&amp;#039;mutex&amp;#039;&amp;#039; für den gegenseitigen Ausschluss (initialisiert auf 1).&lt;br /&gt;
&lt;br /&gt;
Der [[Pseudocode]] für jeden Prozess lautet:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;cpp&amp;quot;&amp;gt;&lt;br /&gt;
void Prozess(int proc_id, int res_id)&lt;br /&gt;
{&lt;br /&gt;
	Ressource_Erwerben(proc_id, res_id);&lt;br /&gt;
&lt;br /&gt;
	&amp;lt;Die Ressource res_id benutzen&amp;gt;;&lt;br /&gt;
&lt;br /&gt;
	Ressource_Freigeben(proc_id, res_id);&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Die [[Funktion (Informatik)|Funktionen]] für das Erwerben und Freigeben der Ressource sind wie folgt aufgebaut:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;cpp&amp;quot;&amp;gt;&lt;br /&gt;
void Ressource_Erwerben(int proc_id, int res_id)&lt;br /&gt;
{&lt;br /&gt;
	P(mutex);&lt;br /&gt;
&lt;br /&gt;
	if (&amp;lt;die Bedingung für den Zugriff auf res_id ist nicht erfüllt für proc_id&amp;gt;)&lt;br /&gt;
	{&lt;br /&gt;
		&amp;lt;erfassen, dass proc_id für res_id blockiert ist&amp;gt;;&lt;br /&gt;
		V(mutex);&lt;br /&gt;
		P(priv[proc_id]);&lt;br /&gt;
		&amp;lt;erfassen, dass proc_id für res_id nicht mehr blockiert ist&amp;gt;;&lt;br /&gt;
	}&lt;br /&gt;
&lt;br /&gt;
	&amp;lt;erfassen, dass proc_id auf res_id zugreift&amp;gt;;&lt;br /&gt;
&lt;br /&gt;
	Staffelstab_Weitergeben(proc_id, res_id); // siehe unten&lt;br /&gt;
}&lt;br /&gt;
&lt;br /&gt;
void Ressource_Freigeben(int proc_id, int res_id)&lt;br /&gt;
{&lt;br /&gt;
	P(mutex);&lt;br /&gt;
&lt;br /&gt;
	&amp;lt;erfassen, dass proc_id auf res_id nicht mehr zugreift&amp;gt;;&lt;br /&gt;
&lt;br /&gt;
	Staffelstab_Weitergeben(proc_id, res_id); // siehe unten&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;Beide [[Operation (Informatik)|Operationen]] verwenden die [[Funktion (Informatik)|Funktion]] &amp;#039;&amp;#039;Staffelstab_Weitergeben&amp;#039;&amp;#039;:&amp;lt;syntaxhighlight lang=&amp;quot;cpp&amp;quot;&amp;gt;&lt;br /&gt;
void Staffelstab_Weitergeben(int proc_id, int res_id)&lt;br /&gt;
{&lt;br /&gt;
	if &amp;lt;die Bedingung für den Zugriff auf res_id ist von mindestens einem blockierten Prozess erfüllt&amp;gt;&lt;br /&gt;
	{&lt;br /&gt;
		int p = &amp;lt;Wahl eines blockierten Prozesses um ihn zu reaktivieren&amp;gt;;&lt;br /&gt;
		V(priv[p]);&lt;br /&gt;
	}&lt;br /&gt;
	else&lt;br /&gt;
	{&lt;br /&gt;
		V(mutex);&lt;br /&gt;
	}&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
&amp;#039;&amp;#039;&amp;#039;Anmerkungen:&amp;#039;&amp;#039;&amp;#039;&lt;br /&gt;
&lt;br /&gt;
Das Pattern wird &amp;#039;&amp;#039;Passing the Baton&amp;#039;&amp;#039; genannt, weil ein [[Prozess (Informatik)|Prozess]], der die Ressource freigibt, sowie ein frisch reaktivierter Prozess höchstens einen anderen angehaltenen Prozess aktiviert, d. h. sozusagen &amp;quot;den Staffelstab an ihn weitergibt&amp;quot;. Die &amp;#039;&amp;#039;mutex&amp;#039;&amp;#039; Semaphore wird nur dann freigegeben, wenn ein Prozess sich selbst blockiert (während der [[Funktion (Informatik)|Funktion]] &amp;#039;&amp;#039;Ressource_Erwerben&amp;#039;&amp;#039;) oder wenn es innerhalb der [[Operation (Informatik)|Operation]] &amp;#039;&amp;#039;Staffelstab_Weitergeben&amp;#039;&amp;#039; nicht möglich ist, einen anderen blockierten Prozess zu reaktivieren.&lt;br /&gt;
&lt;br /&gt;
== Implementierung ==&lt;br /&gt;
&lt;br /&gt;
Eine Implementierung der Semaphormechanismen ist konzeptionell im Betriebssystem anzusiedeln, da sie eng mit der Prozessverwaltung zusammenarbeiten muss. Eine Realisierung der Unteilbarkeit auf Monoprozessorsystemen kann dann z.&amp;amp;nbsp;B. mittels [[Interrupt|Unterbrechungssperre]] erfolgen. Im Fall von Multiprozessorsystemen ist eine Klammerung der Anweisungsfolgen der Semaphoroperationen durch [[Spinlock]]s erforderlich. Eine Realisierung durch das Betriebssystem ermöglicht ferner, dass mehrere Prozesse mit ihren eigentlich [[disjunkt]]en [[Adressraum|Adressräumen]] einen Semaphor gemeinsam nutzen können.&lt;br /&gt;
&lt;br /&gt;
Werden Semaphore von einem Thread-Paket, das im Benutzeradressraum läuft (User-Level-Threads), angeboten, so gestaltet sich die Realisierung der Unteilbarkeit aufwändiger. Sie muss beliebige Unterbrechungen berücksichtigen und hängt davon ab, welche Informationen über User-Level-Threads im Kern vorliegen.&lt;br /&gt;
&lt;br /&gt;
== Verwandte Themen ==&lt;br /&gt;
* [[Mutex]] – Oberbegriff für Verfahren, die wechselseitigen Ausschluss von Datenzugriffen ermöglichen.&lt;br /&gt;
* [[Monitor (Informatik)|Monitor]] – ein programmiersprachliches Konzept zur Prozesssynchronisation.&lt;br /&gt;
* [[Jacketing]] – die Möglichkeit, einen blockierenden Systemaufruf zu umgehen.&lt;br /&gt;
* [[Bolt-Variable]] – Variante des Semaphors zur flexibleren Realisierung eines [[Read-Write-Lock]].&lt;br /&gt;
&lt;br /&gt;
== Literatur ==&lt;br /&gt;
* {{Literatur |Autor=Andrew S. Tanenbaum |Titel=Moderne Betriebssysteme |Reihe=Pearson Studium – IT |Auflage=Dritte aktualisierte |Verlag=Addison-Wesley Verlag |Datum=2009 |ISBN=978-3-8273-7342-7 |Seiten=1248 |Sprache=en |Originaltitel=Modern Operating Systems}}&lt;br /&gt;
* {{Literatur |Autor=James H. Anderson, Yong-Jik Kim, Ted Herman |Titel=Shared-memory mutual exclusion: major research trends since 1986 |Sammelwerk=Distrib. Comput. |Band=16 |Nummer=2–3 |Verlag=Springer-Verlag |Ort=London UK |Datum=2003-09 |ISSN=0178-2770 |Seiten=75–110 |DOI=10.1007/s00446-003-0088-6}}&lt;br /&gt;
* {{Literatur |Autor=M. Raynal, D. Beeson |Titel=Algorithms for mutual exclusion |Verlag=MIT Press |Ort=Cambridge MA |Datum=1986 |ISBN=0-262-18119-3}}&lt;br /&gt;
&lt;br /&gt;
== Weblinks ==&lt;br /&gt;
* [http://bxr.su/FreeBSD/sys/kern/kern_sema.c#28 Implementierung von Semaphoren in FreeBSD, Kernel]&lt;br /&gt;
* [http://bxr.su/FreeBSD/lib/libc/gen/sem_new.c#51 Implementierung von Semaphoren in FreeBSD, API]&lt;br /&gt;
* [http://bxr.su/FreeBSD/include/semaphore.h#30 Schnittstellenspezifikation bzw. API über Semaphoren in FreeBSD]&lt;br /&gt;
* [https://www.freebsd.org/cgi/man.cgi?query=sem_init&amp;amp;sektion=3&amp;amp;manpath=FreeBSD+10.2-RELEASE sem_init(3)]&lt;br /&gt;
* [https://www.freebsd.org/cgi/man.cgi?query=sem_post&amp;amp;sektion=3&amp;amp;manpath=FreeBSD+10.2-RELEASE sem_post(3)]&lt;br /&gt;
* [https://www.freebsd.org/cgi/man.cgi?query=sem_trywait&amp;amp;sektion=3&amp;amp;manpath=FreeBSD+10.2-RELEASE sem_wait(3)]&lt;br /&gt;
* [https://www.freebsd.org/cgi/man.cgi?query=sem_getvalue&amp;amp;sektion=3&amp;amp;manpath=FreeBSD+10.2-RELEASE sem_getvalue(3)]&lt;br /&gt;
* [https://www.freebsd.org/cgi/man.cgi?query=sem_destroy&amp;amp;sektion=3&amp;amp;manpath=FreeBSD+10.2-RELEASE sem_destroy(3)]&lt;br /&gt;
* [https://www.freebsd.org/cgi/man.cgi?query=sem&amp;amp;manpath=FreeBSD+10.2-RELEASE sem(4)]&lt;br /&gt;
&lt;br /&gt;
== Einzelnachweise ==&lt;br /&gt;
&amp;lt;references /&amp;gt;&lt;br /&gt;
&lt;br /&gt;
[[Kategorie:Datenstruktur]]&lt;br /&gt;
[[Kategorie:Parallelverarbeitung]]&lt;br /&gt;
[[Kategorie:Betriebssystemtheorie]]&lt;/div&gt;</summary>
		<author><name>~2026-79915-4</name></author>
	</entry>
</feed>