<?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=Graham_Scan</id>
	<title>Graham Scan - 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=Graham_Scan"/>
	<link rel="alternate" type="text/html" href="https://wiki-de.moshellshocker.dns64.de/index.php?title=Graham_Scan&amp;action=history"/>
	<updated>2026-05-21T12:39:45Z</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=Graham_Scan&amp;diff=739115&amp;oldid=prev</id>
		<title>imported&gt;Wittiko: jpg durch neue svg ersetzt</title>
		<link rel="alternate" type="text/html" href="https://wiki-de.moshellshocker.dns64.de/index.php?title=Graham_Scan&amp;diff=739115&amp;oldid=prev"/>
		<updated>2025-11-14T17:40:36Z</updated>

		<summary type="html">&lt;p&gt;jpg durch neue svg ersetzt&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Neue Seite&lt;/b&gt;&lt;/p&gt;&lt;div&gt;[[Datei:GrahamScanDemo.gif|mini|Animation des Graham Scan Algorithmus]]&lt;br /&gt;
Der &amp;#039;&amp;#039;&amp;#039;Graham Scan&amp;#039;&amp;#039;&amp;#039; (nach [[Ronald Graham]] 1972) ist ein effizienter [[Algorithmus]] zur Berechnung der [[Konvexe Hülle|konvexen Hülle]] einer endlichen Menge von Punkten in der Ebene. Bei &amp;lt;math&amp;gt;n&amp;lt;/math&amp;gt; Punkten liegt seine [[asymptotische Laufzeit]] in &amp;lt;math&amp;gt;\mathcal O(n \cdot \log n)&amp;lt;/math&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
== Beschreibung ==&lt;br /&gt;
=== Vorbereitung ===&lt;br /&gt;
[[Datei:Anglesort.svg|thumb|hochkant=1.2|Sortierung einer Punktmenge nach Winkel mit Bezugspunkt &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt;.]]&lt;br /&gt;
Sei &amp;lt;math&amp;gt;S = \{P\}&amp;lt;/math&amp;gt; eine endliche Punktmenge. Der Algorithmus beginnt mit einem Punkt der Menge, welcher garantiert ein Eckpunkt der konvexen Hülle ist. Man sucht sich dazu den Punkt mit der kleinsten [[Ordinate]]. Sind dies mehrere, so sucht man sich aus diesen Punkten den mit der kleinsten [[Abszisse]] aus ([[lexikographisch]]e Suche). Diese Suche kann in &amp;lt;math&amp;gt;O(n)&amp;lt;/math&amp;gt; Schritten durchgeführt werden. Nachdem der Startpunkt &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; gefunden wurde, sortiert der Algorithmus die restlichen Punkte &amp;lt;math&amp;gt;P&amp;lt;/math&amp;gt; in &amp;lt;math&amp;gt;S&amp;lt;/math&amp;gt; nach aufsteigendem Winkel zwischen &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; → &amp;lt;math&amp;gt;P&amp;lt;/math&amp;gt; und der x-Achse gegen den Uhrzeigersinn. Haben dabei zwei Punkte den gleichen Winkel (d.&amp;amp;nbsp;h. liegen mit &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; auf einer Linie, sind kollinear mit &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt;), so wird der Punkt, welcher näher an &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; liegt, verworfen.&lt;br /&gt;
&lt;br /&gt;
== Hilfsfunktion ==&lt;br /&gt;
[[Datei:Graham_Scan.svg|frame|right|PAB und ABC sind positiv orientierte Dreiecke, BCD ist negativ orientiert. Oder anders, C liegt links von AB, D liegt rechts von BC. Der Stack wird bis [P,A,B,C] aufgebaut, im nächsten Schritt wird C zugunsten von D entfernt.]]&lt;br /&gt;
In der nachfolgenden Rechnung muss wiederholt entschieden werden, ob drei Punkte &amp;lt;math&amp;gt;A=(x_A,y_A)&amp;lt;/math&amp;gt;, &amp;lt;math&amp;gt;B=(x_B,y_B)&amp;lt;/math&amp;gt;, &amp;lt;math&amp;gt;C=(x_C,y_C)&amp;lt;/math&amp;gt; in der Ebene ein positiv orientiertes Dreieck bilden. Äquivalente Formulierungen dafür sind, dass der Streckenzug &amp;lt;math&amp;gt;ABC&amp;lt;/math&amp;gt; einen Knick nach links hat oder dass der Punkt &amp;lt;math&amp;gt;C&amp;lt;/math&amp;gt; links der Strecke von &amp;lt;math&amp;gt;A&amp;lt;/math&amp;gt; nach &amp;lt;math&amp;gt;B&amp;lt;/math&amp;gt; bzw. der Punkt &amp;lt;math&amp;gt;B&amp;lt;/math&amp;gt; rechts von der Strecke von &amp;lt;math&amp;gt;A&amp;lt;/math&amp;gt; nach &amp;lt;math&amp;gt;C&amp;lt;/math&amp;gt; liegt.&lt;br /&gt;
&lt;br /&gt;
Diese Aufgabe kann man durch Bestimmen aller relevanten Winkel lösen, oder einfacher durch die Berechnung einer [[Determinante]] &amp;lt;math&amp;gt;T(A, B, C)&amp;lt;/math&amp;gt;, diese liefert das gewünschte Ergebnis mit weniger Rechenaufwand (fünf Subtraktionen, zwei Multiplikationen) und genauer. Das Ergebnis bleibt für rationale Koordinaten im [[Rationale Zahl|rationalen Zahlenbereich]], welcher ohne Verlust von Genauigkeit im Computer abgebildet werden kann. Das Ergebnis wird über den folgenden Ausdruck berechnet.&lt;br /&gt;
&lt;br /&gt;
:&amp;lt;math&amp;gt;\begin{align}&lt;br /&gt;
T(A, B, C) &amp;amp;= \begin{vmatrix}&lt;br /&gt;
1&amp;amp;x_A&amp;amp;y_A\\&lt;br /&gt;
1&amp;amp;x_B&amp;amp;y_B\\&lt;br /&gt;
1&amp;amp;x_C&amp;amp;y_C&lt;br /&gt;
\end{vmatrix}&lt;br /&gt;
=&lt;br /&gt;
\begin{vmatrix}&lt;br /&gt;
1&amp;amp;x_A    &amp;amp;    y_A\\&lt;br /&gt;
0&amp;amp;x_B-x_A&amp;amp;y_B-y_A\\&lt;br /&gt;
0&amp;amp;x_C-x_A&amp;amp;y_C-y_A&lt;br /&gt;
\end{vmatrix}&lt;br /&gt;
=&lt;br /&gt;
\begin{vmatrix}&lt;br /&gt;
x_B-x_A&amp;amp;y_B-y_A\\&lt;br /&gt;
x_C-x_A&amp;amp;y_C-y_A&lt;br /&gt;
\end{vmatrix}&lt;br /&gt;
\\&amp;amp;=&lt;br /&gt;
(x_B - x_A)(y_C - y_A) - (x_C - x_A)(y_B - y_A) \\&lt;br /&gt;
 &amp;amp;=&lt;br /&gt;
\begin{cases}&lt;br /&gt;
  &amp;lt; 0, &amp;amp; \text{wenn }C\text{ ist rechts von }\overrightarrow{AB} \\&lt;br /&gt;
  = 0, &amp;amp; \text{wenn }C\text{ ist auf }\overrightarrow{AB} \\&lt;br /&gt;
  &amp;gt; 0, &amp;amp; \text{wenn }C\text{ ist links von }\overrightarrow{AB}&lt;br /&gt;
\end{cases}&lt;br /&gt;
\end{align}&amp;lt;/math&amp;gt;&lt;br /&gt;
&lt;br /&gt;
=== Berechnung ===&lt;br /&gt;
[[Datei:sortedpointset.jpg|thumb|310px|Nach Winkel sortierte Punktmenge – &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; wird als Startpunkt gewählt, da er den kleinsten Ordinatenwert hat. &amp;lt;math&amp;gt;P_1&amp;lt;/math&amp;gt; fällt heraus, da er mit &amp;lt;math&amp;gt;P_2&amp;lt;/math&amp;gt; und &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; kollinear ist.]]&lt;br /&gt;
&lt;br /&gt;
&amp;lt;math&amp;gt;S&amp;lt;/math&amp;gt; sei nun die sortierte Punktmenge. Als Nächstes läuft man alle Punkte in &amp;lt;math&amp;gt;S&amp;lt;/math&amp;gt; durch und prüft, ob diese Eckpunkte der konvexen Hülle sind. Es wird ein [[Stapelspeicher]] (Stack) angelegt, auf welchem sich alle Eckpunkte der konvexen Hülle für alle bereits abgearbeiteten Punkte befinden. Zu Beginn liegen &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; und &amp;lt;math&amp;gt;P_1&amp;lt;/math&amp;gt; auf dem Stapel. Im &amp;lt;math&amp;gt;k&amp;lt;/math&amp;gt;-ten Schritt wird &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; zur Betrachtung herangezogen und berechnet, wie er die vorherige konvexe Hülle verändert. Aufgrund der Sortierung liegt &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; immer außerhalb der Hülle der vorherigen Punkte &amp;lt;math&amp;gt;P_i&amp;lt;/math&amp;gt; mit &amp;lt;math&amp;gt;i &amp;lt; k&amp;lt;/math&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
Durch das Hinzufügen des Punktes kann es vorkommen, dass bereits auf dem Stapel liegende Punkte nicht mehr zur neuen konvexen Hülle gehören. Diese Punkte müssen mittels der „pop“ Operation vom Stapel entfernt werden. Ob ein Punkt noch zur konvexen Hülle gehört oder nicht ermittelt man, indem man berechnet, ob &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; links oder rechts des Vektors PT2→PT1 liegt (PT1 = oberstes Element des Stapels, PT2 = Element direkt unter PT1). Liegt &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; links, so bleibt PT1 weiterhin auf dem Stapel und &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; wird mit „push“ auf dem Stapel abgelegt, liegt &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; rechts, so wird PT1 von der neuen konvexen Hülle verschluckt, vom Stapel entfernt und die nächsten beiden oberen Punkte untersucht.&lt;br /&gt;
&lt;br /&gt;
Dieser Test wird solange durchgeführt, bis &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; links des [[Vektor]]s PT2→PT1 oder nur noch &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; und ein weiterer Punkt auf dem Stapel liegt. In beiden Fällen wird dann &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; auf dem Stapel abgelegt und mit dem nächsten Punkt &amp;lt;math&amp;gt;P_{k+1}&amp;lt;/math&amp;gt; weitergerechnet. Die folgende Abbildung zeigt ein Beispiel, in welchem alle Fälle des eben beschriebenen Tests auftreten.&lt;br /&gt;
&lt;br /&gt;
[[Datei:testv.jpg|thumb|310px|Beispiel – Graham Scan Algorithmus]]&lt;br /&gt;
&lt;br /&gt;
In nebenstehender Abbildung werden zunächst die Punkte Pt4, Pt3, Pt2 und &amp;lt;math&amp;gt;P_{k-1}&amp;lt;/math&amp;gt; auf den Stack gelegt. Zu jedem Zeitpunkt bilden die Punkte auf dem Stack ein konvexes Polygon (gestrichelte Linien). Erst als &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; hinzukommen soll, fallen &amp;lt;math&amp;gt;P_{k-1}&amp;lt;/math&amp;gt; und Pt2 wieder raus, da sie zusammen mit &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; nicht konvex sind. Die konvexe Hülle dieser Punktmenge besteht aus &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt;, Pt4, Pt3 und &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt;. &amp;lt;math&amp;gt;P_0&amp;lt;/math&amp;gt; liegt dabei auf dem Stack ganz unten und &amp;lt;math&amp;gt;P_k&amp;lt;/math&amp;gt; ganz oben. Die Punkte des gesuchten konvexen Polygons können mit „pop“ im Uhrzeigersinn vom Stapel geholt werden.&lt;br /&gt;
&lt;br /&gt;
=== Anmerkung ===&lt;br /&gt;
Die Anzahl der „push“ und „pop“ Operationen übersteigt die obere Grenze von &amp;lt;math&amp;gt;2n&amp;lt;/math&amp;gt; (&amp;lt;math&amp;gt;n&amp;lt;/math&amp;gt; = Anzahl der Punkte in der Eingabemenge) nicht. Die Berechnung ist also &amp;lt;math&amp;gt;\mathcal O(n)&amp;lt;/math&amp;gt;. Die Sortierung der Punkte nach Winkel kann mit jedem beliebigen Sortieralgorithmus durchgeführt werden, z.&amp;amp;nbsp;B. [[Mergesort]]. Dieser hat eine asymptotische [[Zeitkomplexität|Laufzeit]] von &amp;lt;math&amp;gt;\mathcal O(n \log n)&amp;lt;/math&amp;gt;. Das bedeutet, dass die Laufzeit des Algorithmus durch die Sortierung vorgegeben ist, da &amp;lt;math&amp;gt;\mathcal O(n) + \mathcal O(n \log n) = \mathcal O(n \log n)&amp;lt;/math&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
== Pseudocode ==&lt;br /&gt;
=== Unter Nutzung eines Stacks ===&lt;br /&gt;
 &amp;#039;&amp;#039;&amp;#039;Funktion&amp;#039;&amp;#039;&amp;#039; GrahamScan&lt;br /&gt;
   Eingabe: Punktemenge S = {P}&lt;br /&gt;
   Ausgabe: konvexe Hülle von S&lt;br /&gt;
   &amp;#039;&amp;#039;&amp;#039;Beginn&amp;#039;&amp;#039;&amp;#039; Funktion&lt;br /&gt;
     Sei S die nach dem Winkel zu P&amp;lt;sub&amp;gt;0&amp;lt;/sub&amp;gt; sortierte Punktemenge&lt;br /&gt;
     PUSH(P&amp;lt;sub&amp;gt;0&amp;lt;/sub&amp;gt;)&lt;br /&gt;
     PUSH(P&amp;lt;sub&amp;gt;1&amp;lt;/sub&amp;gt;)&lt;br /&gt;
     i := 2&lt;br /&gt;
     n := Anzahl der Punkte in S&lt;br /&gt;
     &lt;br /&gt;
     &amp;#039;&amp;#039;&amp;#039;Solange&amp;#039;&amp;#039;&amp;#039; i &amp;lt; n, führe aus:&lt;br /&gt;
       Sei Pt&amp;lt;sub&amp;gt;1&amp;lt;/sub&amp;gt; der oberste Punkt auf dem Stack&lt;br /&gt;
       Sei Pt&amp;lt;sub&amp;gt;2&amp;lt;/sub&amp;gt; der zweitoberste Punkt auf dem Stack&lt;br /&gt;
       &amp;#039;&amp;#039;&amp;#039;Wenn&amp;#039;&amp;#039;&amp;#039; S&amp;lt;sub&amp;gt;i&amp;lt;/sub&amp;gt; links des Vektors Pt&amp;lt;sub&amp;gt;2&amp;lt;/sub&amp;gt;→Pt&amp;lt;sub&amp;gt;1&amp;lt;/sub&amp;gt; liegt &amp;#039;&amp;#039;&amp;#039;oder&amp;#039;&amp;#039;&amp;#039; Stack enthält 2 Elemente, &amp;#039;&amp;#039;&amp;#039;dann&amp;#039;&amp;#039;&amp;#039; führe aus:&lt;br /&gt;
         PUSH(S&amp;lt;sub&amp;gt;i&amp;lt;/sub&amp;gt;)&lt;br /&gt;
         i := i + 1&lt;br /&gt;
       &amp;#039;&amp;#039;&amp;#039;Ansonsten&amp;#039;&amp;#039;&amp;#039; führe aus:&lt;br /&gt;
         POP(Pt&amp;lt;sub&amp;gt;1&amp;lt;/sub&amp;gt;)&lt;br /&gt;
       &amp;#039;&amp;#039;&amp;#039;Ende&amp;#039;&amp;#039;&amp;#039; Bedingung&lt;br /&gt;
       &lt;br /&gt;
     &amp;#039;&amp;#039;&amp;#039;Ende&amp;#039;&amp;#039;&amp;#039; Schleife&lt;br /&gt;
     &lt;br /&gt;
   &amp;#039;&amp;#039;&amp;#039;Ende&amp;#039;&amp;#039;&amp;#039; Funktion&lt;br /&gt;
&lt;br /&gt;
=== Ohne Nutzung eines Stacks ===&lt;br /&gt;
 &amp;#039;&amp;#039;&amp;#039;Funktion&amp;#039;&amp;#039;&amp;#039; GrahamScan&lt;br /&gt;
   Eingabe: Punktemenge S = {P}&lt;br /&gt;
   Ausgabe: konvexe Hülle von S&lt;br /&gt;
   &amp;#039;&amp;#039;&amp;#039;Beginn&amp;#039;&amp;#039;&amp;#039; Funktion&lt;br /&gt;
     Sei S die nach dem Winkel zu P&amp;lt;sub&amp;gt;0&amp;lt;/sub&amp;gt; sortierte Punktemenge&lt;br /&gt;
     i := 1&lt;br /&gt;
     &lt;br /&gt;
     &amp;#039;&amp;#039;&amp;#039;Solange&amp;#039;&amp;#039;&amp;#039; i ≤ |S|:&lt;br /&gt;
       &amp;#039;&amp;#039;&amp;#039;Wenn&amp;#039;&amp;#039;&amp;#039; S&amp;lt;sub&amp;gt;i&amp;lt;/sub&amp;gt; rechts des Vektors S&amp;lt;sub&amp;gt;i−1&amp;lt;/sub&amp;gt;→S&amp;lt;sub&amp;gt;i+1&amp;lt;/sub&amp;gt; liegt, &amp;#039;&amp;#039;&amp;#039;dann&amp;#039;&amp;#039;&amp;#039; führe aus:&lt;br /&gt;
         i := i + 1&lt;br /&gt;
       &amp;#039;&amp;#039;&amp;#039;Ansonsten&amp;#039;&amp;#039;&amp;#039; führe aus:&lt;br /&gt;
         Entferne das Element S&amp;lt;sub&amp;gt;i&amp;lt;/sub&amp;gt; aus S&lt;br /&gt;
         i := i - 1&lt;br /&gt;
       &amp;#039;&amp;#039;&amp;#039;Ende&amp;#039;&amp;#039;&amp;#039; Bedingung&lt;br /&gt;
     &lt;br /&gt;
     &amp;#039;&amp;#039;&amp;#039;Ende&amp;#039;&amp;#039;&amp;#039; Schleife&lt;br /&gt;
   &lt;br /&gt;
   &amp;#039;&amp;#039;&amp;#039;Ende&amp;#039;&amp;#039;&amp;#039; Funktion&lt;br /&gt;
&lt;br /&gt;
Im Code sei &amp;lt;code&amp;gt;punkte&amp;lt;/code&amp;gt; ein [[Feld (Datentyp)|Array]] aus Punkten, aus dem man mit &amp;lt;code&amp;gt;punkte[i]&amp;lt;/code&amp;gt; das &amp;lt;code&amp;gt;i&amp;lt;/code&amp;gt;-te Element erhält und welches schon nach dem Winkel zu &amp;lt;code&amp;gt;punkte[0]&amp;lt;/code&amp;gt; sortiert ist. Der Code verändert dieses Array, indem die Elemente gelöscht werden, die nicht zur konvexen Hülle gehören.&lt;br /&gt;
&lt;br /&gt;
== Programmierung ==&lt;br /&gt;
Das folgende Beispiel in der [[Programmiersprache]] [[C-Sharp|C#]] zeigt die Implementierung des Graham Scan Algorithmus. Die Punkte und die [[konvexe Hülle]] werden auf dem Hauptfenster gezeichnet. Das Programm verwendet mehrere [[Klasse (Objektorientierung)|Klassen]]. Die [[Methode (Programmierung)|Methoden]] für den eigentlichen Algorithmus werden in der Klasse &amp;#039;&amp;#039;GrahamScan&amp;#039;&amp;#039; deklariert.&amp;lt;ref&amp;gt;Rosetta Code: [https://rosettacode.org/wiki/Convex_hull Convex hull]&amp;lt;/ref&amp;gt;&amp;lt;ref&amp;gt;GeeksforGeeks: [https://www.geeksforgeeks.org/convex-hull-set-2-graham-scan/ Convex Hull]&amp;lt;/ref&amp;gt;&lt;br /&gt;
{| class=&amp;quot;wikitable left mw-collapsible mw-collapsed font-size: 105.3%;&amp;quot;&lt;br /&gt;
|style=&amp;quot;text-align:left; font-size: 95%;&amp;quot;| &amp;#039;&amp;#039;&amp;#039;Code-Schnipsel&amp;#039;&amp;#039;&amp;#039;&amp;amp;nbsp;&amp;amp;nbsp;&lt;br /&gt;
|-&lt;br /&gt;
|&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;c#&amp;quot;&amp;gt;&lt;br /&gt;
using System;&lt;br /&gt;
using System.Collections.Generic;&lt;br /&gt;
using System.Drawing;&lt;br /&gt;
using System.Windows.Forms;&lt;br /&gt;
&lt;br /&gt;
// Klasse, die die Methoden für den Algorithmus Graham Scan deklariert&lt;br /&gt;
class GrahamScan&lt;br /&gt;
{&lt;br /&gt;
	// Diese Methode gibt das zweitoberste Element im Stack der Punkte zurück&lt;br /&gt;
	private PointF GetNextToTopElement(Stack&amp;lt;PointF&amp;gt; points)&lt;br /&gt;
	{&lt;br /&gt;
		PointF topElement = points.Peek();&lt;br /&gt;
		points.Pop();&lt;br /&gt;
		PointF nextToTopElement = points.Peek();&lt;br /&gt;
		points.Push(topElement);&lt;br /&gt;
		return nextToTopElement;&lt;br /&gt;
	}&lt;br /&gt;
	&lt;br /&gt;
	// Diese Methode gibt den Abstand der zwei Punkte zurück&lt;br /&gt;
	private float GetDistance(PointF point1, PointF point2)&lt;br /&gt;
	{&lt;br /&gt;
		return (point1.X - point2.X) * (point1.X - point2.X) + (point1.Y - point2.Y) * (point1.Y - point2.Y);&lt;br /&gt;
	}&lt;br /&gt;
	&lt;br /&gt;
	// Diese Methode bestimmt die Orientierung des drei Punkte. Wenn die Punkte im Uhrzeigersinn sind, wird der Wert 1 zurückgegeben. Wenn die Punkte im Gegenuhrzeigersinn sind, wird der Wert -1 zurückgegeben. Wenn die Punkte kollinear sind, wird der Wert 0 zurückgegeben.&lt;br /&gt;
	private int GetOrientation(PointF point1, PointF point2, PointF point3)&lt;br /&gt;
	{&lt;br /&gt;
		float area = (point2.Y - point1.Y) * (point3.X - point2.X) - (point2.X - point1.X) * (point3.Y - point2.Y);&lt;br /&gt;
		if (area &amp;gt; 0) // Wenn das Vorzeichen positiv ist, sind die Punkte im Uhrzeigersinn&lt;br /&gt;
		{&lt;br /&gt;
			return 1;&lt;br /&gt;
		}&lt;br /&gt;
		if (area &amp;lt; 0) // Wenn das Vorzeichen negativ ist, sind die Punkte im Gegenuhrzeigersinn&lt;br /&gt;
		{&lt;br /&gt;
			return -1;&lt;br /&gt;
		}&lt;br /&gt;
		return 0; // Wenn der Flächeninhalt gleich 0 ist, sind die Punkte kollinear&lt;br /&gt;
	}&lt;br /&gt;
	&lt;br /&gt;
	// Diese Klasse implementiert eine Vergleichsmethode für das Sortieren der Punkte&lt;br /&gt;
	class PointComparer : Comparer&amp;lt;PointF&amp;gt;&lt;br /&gt;
	{&lt;br /&gt;
		public PointF center;&lt;br /&gt;
		public GrahamScan grahamScan;&lt;br /&gt;
		&lt;br /&gt;
		// Vergleichsmethode, die die Punkte in Bezug auf den Punkt center sortiert&lt;br /&gt;
		public override int Compare(PointF point1, PointF point2)&lt;br /&gt;
		{&lt;br /&gt;
			int orientation = grahamScan.GetOrientation(center, point1, point2);&lt;br /&gt;
			if (orientation == 0)&lt;br /&gt;
			{&lt;br /&gt;
				if (grahamScan.GetDistance(center, point2) &amp;gt;= grahamScan.GetDistance(center, point1))&lt;br /&gt;
				{&lt;br /&gt;
					return -1;&lt;br /&gt;
				}&lt;br /&gt;
				return 1;&lt;br /&gt;
			}&lt;br /&gt;
			return orientation;&lt;br /&gt;
		}&lt;br /&gt;
	}&lt;br /&gt;
	&lt;br /&gt;
	// Diese Methode gibt eine Liste der Punkte der konvexen Hülle zurück&lt;br /&gt;
	public List&amp;lt;PointF&amp;gt; GetConvexHull(List&amp;lt;PointF&amp;gt; points)&lt;br /&gt;
	{&lt;br /&gt;
		List&amp;lt;PointF&amp;gt; convexHull = new List&amp;lt;PointF&amp;gt;(); // Initialisiert die Liste der Punkte der konvexen Hülle&lt;br /&gt;
		float minimumY = points[0].Y;&lt;br /&gt;
		int indexOfMinimum = 0;&lt;br /&gt;
		int numberOfpoints = points.Count;&lt;br /&gt;
		// Diese for-Schleife durchläuft die verbleibenden Punkte und ermittelt den untersten Punkt und bei Gleichheit den Punkt ganz links&lt;br /&gt;
		for (int i = 1; i &amp;lt; numberOfpoints; i++)&lt;br /&gt;
		{&lt;br /&gt;
			float y = points[i].Y;&lt;br /&gt;
			if (y &amp;lt; minimumY || minimumY == y &amp;amp;&amp;amp; points[i].X &amp;lt; points[indexOfMinimum].X)&lt;br /&gt;
			{&lt;br /&gt;
				minimumY = points[i].Y;&lt;br /&gt;
				indexOfMinimum = i;&lt;br /&gt;
			}&lt;br /&gt;
		}&lt;br /&gt;
		// Setzt den untersten Punkt an die erste Position&lt;br /&gt;
		PointF point = points[0];&lt;br /&gt;
		points[0] = points[indexOfMinimum];&lt;br /&gt;
		points[indexOfMinimum] = point;&lt;br /&gt;
		// Initialisiert das Objekt, das die Vergleichsmethode bereitstellt&lt;br /&gt;
		PointComparer pointComparer = new PointComparer();&lt;br /&gt;
		pointComparer.center = points[0];&lt;br /&gt;
		pointComparer.grahamScan = this;&lt;br /&gt;
		points.Sort(pointComparer); // Sortiert die Punkte in Bezug auf den ersten Punkt&lt;br /&gt;
		// Wenn zwei oder mehr Punkte den gleichen Winkel mit dem Punkt point bilden, werden alle entfernt außer dem Punkt, der am weitesten vom Punkt point entfernt ist&lt;br /&gt;
		int index = 1;&lt;br /&gt;
		for (int i = 1; i &amp;lt; numberOfpoints; i++)&lt;br /&gt;
		{&lt;br /&gt;
			// Entfernt jeweils das Element mit Index i, solange die Punkte der gegebenen Elemente kollinear sind&lt;br /&gt;
			while (i &amp;lt; numberOfpoints - 1 &amp;amp;&amp;amp; GetOrientation(pointComparer.center, points[i], points[i + 1]) == 0)&lt;br /&gt;
			{&lt;br /&gt;
				i++;&lt;br /&gt;
			}&lt;br /&gt;
			points[index] = points[i];&lt;br /&gt;
			index++;&lt;br /&gt;
		}&lt;br /&gt;
		if (index &amp;lt; 3) // Wenn weniger als 3 Punkte vorhanden sind, wird eine leere Liste zurückgegeben&lt;br /&gt;
		{&lt;br /&gt;
			return convexHull;&lt;br /&gt;
		}&lt;br /&gt;
		// Erzeugt einen Stack und fügt die ersten 3 Punkte hinzu&lt;br /&gt;
		Stack&amp;lt;PointF&amp;gt; pointStack = new Stack&amp;lt;PointF&amp;gt;();&lt;br /&gt;
		pointStack.Push(points[0]);&lt;br /&gt;
		pointStack.Push(points[1]);&lt;br /&gt;
		pointStack.Push(points[2]);&lt;br /&gt;
		// Diese for-Schleife durchläuft die verbleibenden Punkte&lt;br /&gt;
		for (int i = 3; i &amp;lt; index; i++)&lt;br /&gt;
		{&lt;br /&gt;
			// Entfernt jeweils das oberste Element des Stack, solange die Punkte der gegebenen Elemente im Uhrzeigersinn sind&lt;br /&gt;
			while (pointStack.Count &amp;gt; 1 &amp;amp;&amp;amp; GetOrientation(GetNextToTopElement(pointStack), pointStack.Peek(), points[i]) != -1)&lt;br /&gt;
			{&lt;br /&gt;
				pointStack.Pop();&lt;br /&gt;
			}&lt;br /&gt;
			pointStack.Push(points[i]);&lt;br /&gt;
		}&lt;br /&gt;
		// Fügt die Punkte des Stack der Liste der Punkte der konvexen Hülle hinzu&lt;br /&gt;
		while (pointStack.Count != 0)&lt;br /&gt;
		{&lt;br /&gt;
			convexHull.Add(pointStack.Peek());&lt;br /&gt;
			pointStack.Pop();&lt;br /&gt;
		}&lt;br /&gt;
		return convexHull;&lt;br /&gt;
	}&lt;br /&gt;
}&lt;br /&gt;
&lt;br /&gt;
// Klasse für das Hauptfenster&lt;br /&gt;
public partial class MainForm : Form&lt;br /&gt;
{&lt;br /&gt;
	private Graphics graphics;&lt;br /&gt;
	&lt;br /&gt;
	private List&amp;lt;PointF&amp;gt; points = new List&amp;lt;PointF&amp;gt;(); // Liste der Punkte&lt;br /&gt;
	private List&amp;lt;PointF&amp;gt; convexHull = new List&amp;lt;PointF&amp;gt;(); // Liste der Punkte der konvexen Hülle&lt;br /&gt;
	private double x1, y1, x2, y2;&lt;br /&gt;
	&lt;br /&gt;
	public MainForm()&lt;br /&gt;
	{&lt;br /&gt;
		x1 = 100; y1 = 100; x2 = 700; y2 = 700; // Setzt die Koordinaten der Eckpunkte der quadratischen Zeichenfläche&lt;br /&gt;
		Random random = new Random(); // Initialisiert den Zufallsgenerator&lt;br /&gt;
		int numberOfPoints = 100;&lt;br /&gt;
		for (int i = 0; i &amp;lt; numberOfPoints; i++) // Diese for-Schleife erzeugt 100 zufällige Punkte innerhalb der quadratischen Zeichenfläche&lt;br /&gt;
		{&lt;br /&gt;
			PointF point = new PointF();&lt;br /&gt;
			point.X = (float)(random.NextDouble() * (x2 - x1) + x1);&lt;br /&gt;
			point.Y = (float)(random.NextDouble() * (y2 - y1) + y1);&lt;br /&gt;
			points.Add(point); // Fügt den Punkt der Liste hinzu&lt;br /&gt;
		}&lt;br /&gt;
		&lt;br /&gt;
		GrahamScan grahamScan = new GrahamScan(); // Erzeugt ein Objekt der Klasse GrahamScan&lt;br /&gt;
		convexHull = grahamScan.GetConvexHull(points); // Aufruf der Methode, die die konvexe Hülle zurückgibt&lt;br /&gt;
		&lt;br /&gt;
		InitializeComponent();&lt;br /&gt;
		Text = &amp;quot;Konvexe Hülle&amp;quot;;&lt;br /&gt;
		Width = 800;&lt;br /&gt;
		Height = 800;&lt;br /&gt;
		graphics = CreateGraphics(); // Erzeugt ein Grafikobjekt für das Zeichnen auf dem Hauptfenster.&lt;br /&gt;
		Paint += OnPaint; // Verknüpft die Ereignisbehandlungsmethode mit dem Paint Ereignis des Hauptfensters.&lt;br /&gt;
	}&lt;br /&gt;
	&lt;br /&gt;
	// Diese Methode wird aufgerufen, wenn das Hauptfenster gezeichnet wird.&lt;br /&gt;
	private void OnPaint(object sender, PaintEventArgs e)&lt;br /&gt;
	{&lt;br /&gt;
		for (int i = 0; i &amp;lt; points.Count; i++)&lt;br /&gt;
		{&lt;br /&gt;
			graphics.FillRectangle(new SolidBrush(Color.FromArgb(0, 0, 0)), points[i].X - 1, points[i].Y - 1, 2, 2); // Zeichnet die Punkte&lt;br /&gt;
		}&lt;br /&gt;
		int numberOfPoints = convexHull.Count;&lt;br /&gt;
		for (int i = 0; i &amp;lt; numberOfPoints; i++)&lt;br /&gt;
		{&lt;br /&gt;
			graphics.DrawLine(new Pen(Color.FromArgb(0, 0, 255)), convexHull[i], convexHull[(i + 1) % numberOfPoints]); // Zeichnet die Kanten der konvexen Hülle&lt;br /&gt;
		}&lt;br /&gt;
	}&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
|}&lt;br /&gt;
&lt;br /&gt;
== Weblinks ==&lt;br /&gt;
* [http://www.informatik.uni-trier.de/~naeher/Professur/PROJECTS/grahams_scan/index.htm Erklärungen samt Visualisierung des Verfahrens]&lt;br /&gt;
* [http://cs.smith.edu/~jorourke/books/compgeom.html Computational Geometry in C (Second Edition)]&lt;br /&gt;
* [http://www.iti.fh-flensburg.de/lang/algorithmen/geo/graham.htm Erklärung und Pseudocode]&lt;br /&gt;
* [https://github.com/njanakiev/graham-scan Animation des Graham Scan Algorithmus]&lt;br /&gt;
&lt;br /&gt;
== Einzelnachweise ==&lt;br /&gt;
&amp;lt;references /&amp;gt;&lt;br /&gt;
&lt;br /&gt;
[[Kategorie:Algorithmische Geometrie]]&lt;/div&gt;</summary>
		<author><name>imported&gt;Wittiko</name></author>
	</entry>
</feed>