Algorithmus von Hopcroft und Karp

Der Algorithmus v​on Hopcroft u​nd Karp (1973 v​on John E. Hopcroft u​nd Richard M. Karp entwickelt) d​ient in d​er Graphentheorie z​ur Bestimmung e​ines maximalen Matchings i​n einem bipartiten Graphen. Er g​eht aus v​on dem Matching, d​ie keine Kanten enthält, u​nd konstruiert d​azu alternierende Pfade zwischen n​och ungepaarten Knoten. Jeder solche Pfad liefert e​ine Vergrößerung (Augmentierung) d​es Matchings u​m eine Kante.

Augmentierende Pfade

Ist zu einem Graphen mit Kantenmenge ein Matching gegeben, so betrachten wir zusammenhängende Teilgraphen, die Bäume sind, also keinen Zyklus enthalten und die bestehen aus

  • einem ungepaarten Knoten als Wurzel,
  • gepaarten Knoten, die sich von der Wurzel aus innerhalb des Baumes erreichen lassen auf alternierenden Pfaden gerader Kantenzahl. Alternierend heißt, dass die Kanten des Pfades abwechselnd zu gehören und nicht zu gehören. Diese Knoten haben also geraden Abstand von der Wurzel.
  • allen Knoten und Kanten entlang der alternierenden Pfade. Dadurch kommen auch Knoten hinzu, die ungeraden Abstand von der Wurzel haben.

Eine Vereinigungsmenge solcher Bäume, d​ie keine gemeinsamen Knoten haben, heißt Wald.

Wenn Knoten und aus zwei verschiedenen Bäumen des Waldes, die jeweils geraden Abstand von ihrer Wurzel haben, durch die Kante verbunden sind, so kann diese Kante nicht zu gehören, denn die Knoten sind ja schon durch je eine andere Kante innerhalb des Baumes gepaart, es sei denn, es handelt sich um die Wurzel, die sowieso ungepaart ist. Der Pfad mit Kantenmenge von der Wurzel des einen Baumes über zur Wurzel des anderen Baumes ist dann ein alternierender Pfad mit ungepaartem Anfangs- und Endknoten. Ein solcher Pfad wird -augmentierender Pfad genannt, denn ist ein Matching, das eine Kante mehr enthält als .

Umgekehrt gilt, dass ein Matching , das mehr Kanten enthält als , einen Teilgraph mit Kantenmenge ergibt, in dem alle Pfade zwischen und alternieren, und von denen mindestens Pfade -augmentierend ohne gemeinsame Knoten sein müssen. ist also genau dann ein maximales Matching, wenn es keinen -augmentierenden Pfad gibt.

Ungarische Wälder

Bei der Definition der betrachteten Wälder wurde bisher nicht vorausgesetzt, dass ein bipartiter Graph vorliegt. In einem bipartiten Graphen gilt aber mehr: Dort liegen die Knoten geraden Abstandes von ihrer Wurzel in oder , je nachdem wo die Wurzel auch liegt. Wenn es dann im Wald keine zwei Knoten und mit wie im letzten Abschnitt gibt und der Wald auch nicht mehr unter Einhaltung der genannten Eigenschaften vergrößert werden kann, heißt er ein ungarischer Wald. Wegen der Bipartitheit lässt sich dann zeigen, dass das Matching genau dann ein maximales Matching ist, wenn es zu ihm einen ungarischen Wald gibt.

Algorithmus

Der folgende Algorithmus ist eine Vorstufe zum Algorithmus von Hopcroft und Karp. Er konstruiert zu einem bipartiten Graphen mit Matching einen Wald mit den genannten Eigenschaften, der

  1. Beginne mit dem Wald, der alle ungepaarten Knoten als Wurzeln enthält, aber keine Kanten.
  2. Suche eine Kante von einem Knoten des Waldes mit geradem Abstand von seiner Wurzel zu einem Knoten , der nicht zum Wald gehört oder geraden Abstand von seiner Wurzel hat. Falls es keinen solchen Knoten mehr gibt, ist der Wald ein ungarischer Wald. Beende den Algorithmus.
  3. Falls geraden Abstand von seiner Wurzel hat, gibt es einen Pfad gerader Länge von einem ungepaarten Knoten nach . Gib den -augmentierenden Pfad von über nach zurück und beende den Algorithmus.
  4. Falls nicht zum Wald gehört, ist gepaart, etwa : Füge die Knoten und sowie die Kanten und zum Wald hinzu und gehe zurück zu Schritt 2.

Zu Beginn wird der Algorithmus mit ausgeführt. Falls er in Schritt 3 mit einem -augmentierenden Pfad endet, wird durch ersetzt und der Algorithmus erneut durchgeführt. Der Fall, dass der Algorithmus in Schritt 2 mit einem ungarischen Wald endet, wobei dann eine maximales Matching ist, muss nach spätestens Durchläufen des Algorithmus eintreten, weil das Matching im anderen Fall jeweils um zwei Knoten vergrößert wird. Die Laufzeit bei einmaliger Durchführung des Algorithmus ist proportional zur Kantenzahl , die Gesamtlaufzeit bei mehrmaliger Durchführung also proportional zum Produkt aus Kanten- und Knotenzahl.

Beispiel

Der bipartite Graph im Beispiel hat 10 Knoten und 10 Kanten. Im linken Teil sind vor der ersten Phase alle Knoten frei, das Matching ist leer. Alle augmentierenden Pfade werden auf eine einzelne Kante zwischen einem Knoten in und einem Knoten in reduziert. Eine Breitensuche wählt zum Beispiel die Kanten , indem sie für jeden Knoten in den freien Knoten in mit dem kleinsten Index auswählt. Diese Menge von Pfaden ist maximal, alle haben die gleiche Länge 1, das erhaltene Matching hat die Größe 4 und es verbleiben zwei freie Knoten, und .

In der zweiten Phase geht es darum, die Pfade mit minimaler Längenzunahme ausgehend vom einzigen freien Knoten von oder vom einzigen freien Knoten von zu finden. Der angegebene Pfad wechselt zwischen schwarzen Kanten außerhalb des Matchings und roten Kanten innerhalb des Matchings. Er hat die Länge 5. Wir können sehen, dass es keinen Pfad der Länge 3 gibt, aber es gibt einen Pfad der Länge 7, nämlich . Das Matching, das sich aus der symmetrischen Differenz des vorherigen Matchings mit dem Weg der Länge 5 ergibt, ergibt im Beispiel das Matching der Größe 5. Es ist maximal, weil es keine freien Knoten mehr gibt.

Gleichzeitige Augmentierung mehrerer Pfade

Die Gesamtlaufzeit des Algorithmus kann verringert werden, wenn mehrere -augmentierende Pfade gleichzeitig betrachtet werden. Es sei die Länge des kürzesten -augmentierenden Pfades. Wir betrachten -augmentierende knotendisjunkte Pfade der Länge , denen sich kein weiterer -augmentierender Pfad der Länge knotendisjunkt hinzufügen lässt. Dann lässt sich zeigen, dass

Der genannte Algorithmus k​ann so z​um Algorithmus v​on Hopcroft u​nd Karp erweitert werden, d​ass er n​icht nur einen augmentierenden Pfad zurückgibt, sondern e​ine Menge augmentierender Pfade w​ie gerade betrachtet. Dazu müssen Schritt 2 u​nd 3 a​ls Breitensuche durchgeführt werden, w​obei die konstruierten Pfade i​m Wald e​rst dann verlängert werden, w​enn keine n​euen Pfade d​er bisherigen Länge m​ehr zu finden sind. Sobald e​in Pfad z​u einem ungepaarten Knoten führt, a​lso ein augmentierender Pfad ist, brauchen k​eine Pfade n​och größerer Länge m​ehr betrachtet z​u werden.

Ist ein maximales Matching und , so liefert der so erweiterte Algorithmus nach Durchläufen ein Matching mit und knotendisjunkte -augmentierende Pfade, deren Länge mindestens ist. Weil keiner der Knoten in zweien dieser Pfade enthalten ist, muss sein, also muss das maximale Matching nach spätestens weiteren Durchläufen erreicht sein. Die Gesamtlaufzeit des Algorithmus von Hopcroft und Karp ist demnach proportional zum Produkt aus Kantenzahl und Quadratwurzel der Knotenzahl.

Vergleich mit anderen Algorithmen

Für dünne Graphen w​eist der Algorithmus v​on Hopcroft u​nd Karp weiterhin d​ie beste Worst-Case-Laufzeit auf. Für dichte Graphen erzielt e​in neuerer Algorithmus e​ine etwas bessere Laufzeit.[1]

Dieser Algorithmus basiert a​uf der Verwendung d​es Goldberg-Tarjan-Algorithmus. Wenn d​as durch diesen Algorithmus erzeugte Matching nahezu optimal ist, w​ird zum Algorithmus v​on Hopcroft u​nd Karp gewechselt. Mehrere Autoren h​aben experimentelle Vergleiche v​on Algorithmen für bipartite Matchings durchgeführt. Ihre Ergebnisse zeigen tendenziell, d​ass der Algorithmus v​on Hopcroft u​nd Karp i​n der Praxis n​icht so g​ut ist w​ie in d​er Theorie. Er w​ird sowohl d​urch einfachere Strategien für d​ie Suche n​ach augmentierenden Pfaden a​ls auch d​urch Push-Relabel-Techniken übertroffen.[2]

Programmierung

Das folgende Beispiel in der Programmiersprache C++ zeigt die Implementierung des Algorithmus von Hopcroft und Karp für einen bipartiten Graphen. Der bipartite Graph wird als Klasse BipartiteGraph deklariert. Die Methode HopcroftKarp verwendet Breitensuche und Tiefensuche. Bei der Ausführung des Programms wird die Methode main verwendet, die die Liste der markierten Knoten auf der Konsole ausgibt.[3]

#include <list>
#include <queue>
#include <iostream>

using namespace std;

#define NIL 0
#define INF INT_MAX

// Deklariert den Datentyp für die Knoten des Graphen
struct Node
{
    int index;
    string value;
    Node* next;
};

// Deklariert die Klasse für den bipartiten Graphen
class BipartiteGraph
{
    int count1, count2; // Anzahl der Knoten auf der linken und rechten Seite des bipartiten Graphen
    list<Node>* adjacencyList; // Anzahl der Knoten auf der linken und rechten Seite des bipartiten Graphen
    Node* pair1, * pair2; int distance[]; // Basiszeiger, die in der Methode HopcroftKarp verwendet werden

// Deklariert die Interfaces für die Methoden
public:
    BipartiteGraph(int count1, int count2); // Konstruktor
    void AddEdge(Node* startNode, Node* targetNode);
    bool BreadthFirstSearch();
    bool DepthFirstSearch(Node* startNode);
    int HopcroftKarp();
};

// Gibt die Matchingzahl des maximalen Matching zurück
int BipartiteGraph::HopcroftKarp()
{
    pair1 = new Node[count1 + 1]; // Speichert ein Paar von Knoten, wobei pair1[u] eine Ecke auf der linken Seite des bipartiten Graphen ist. Wenn nicht vorhanden, wird der Nullzeiger NIL gespeichert.
    pair2 = new Node[count2 + 1]; // Speichert ein Paar von Knoten, wobei pair1[v] eine Ecke auf der rechten Seite des bipartiten Graphen ist. Wenn nicht vorhanden, wird der Nullzeiger NIL gespeichert.
    // Initialisiert die Paare mit dem Nullzeiger NIL
    for (int i = 0; i <= count1; i++)
    {
        (&pair1)[i] = NIL;
    }
    for (int i = 0; i <= count2; i++)
    {
        (&pair2)[i] = NIL;
    }
    int result = 0; // Initialisierung
    while (BreadthFirstSearch())
    {
        for (int i = 1; i <= count1; i++)
        {
            Node* node = &pair1[i];
            if (node == NIL && DepthFirstSearch(node))
            {
                result++;
            }
        }
    }
    return result;
}

// Gibt true zurück, wenn es einen augmentierenden Pfad gibt, sonst false
bool BipartiteGraph::BreadthFirstSearch()
{
    queue<Node> nodeQueue;
    for (int i = 1; i <= count1; i++)
    {
        Node* node = &pair1[i];
        if (node == NIL)
        {
            distance[i] = 0;
            nodeQueue.push(*node);
        }
        else
        {
            distance[i] = INF;
        }
    }
    distance[NIL] = INF;
    while (!nodeQueue.empty())
    {
        int currentIndex = nodeQueue.front().index;
        nodeQueue.pop();
        if (distance[currentIndex] < distance[NIL])
        {
            list<Node>::iterator i;
            for (i = adjacencyList[currentIndex].begin(); i != adjacencyList[currentIndex].end(); ++i)
            {
                Node* node = &pair2[(*i).index];
                int index = (*node).index;
                if (distance[index] == INF)
                {
                    distance[index] = distance[currentIndex] + 1;
                    nodeQueue.push(*node);
                }
            }
        }
    }
    return (distance[NIL] != INF);
}

// Gibt true zurück, wenn es einen augmentierenden Pfad mit startNode gibt, sonst false
bool BipartiteGraph::DepthFirstSearch(Node* startNode)
{
    int startIndex = startNode->index; //
    if (startNode != NIL)
    {
        list<Node>::iterator i;
        for (i = adjacencyList[startIndex].begin(); i != adjacencyList[startIndex].end(); ++i)
        {
            Node j = *i;
            if (distance[pair2[startIndex].index] == distance[startIndex] + 1)
            {
                if (DepthFirstSearch(&pair2[startIndex]))
                {
                    pair2[startIndex] = *startNode;
                    pair1[startIndex] = j;
                    return true;
                }
            }
        }
        distance[startIndex] = INF; // Es gibt keinen augmentierenden Pfad, der mit startNode beginnt.
        return false;
    }
    return true;
}

// Konstruktor mit der Anzahl count1 und count2
BipartiteGraph::BipartiteGraph(int count1, int count2)
{
    this->count1 = count1;
    this->count2 = count2;
    adjacencyList = new list<Node>[count1 + 1];
}

// Diese Methode verbindet die Knoten startNode und targetNode miteinander.
void BipartiteGraph::AddEdge(Node* startNode, Node* targetNode)
{
    adjacencyList[(*startNode).index].push_back(*targetNode);
}

// Hauptmethode, die das Programm ausführt
int main()
{
    // Deklariert und initialisiert 4 Knoten
    Node node1 = Node{ 1, "A", NIL };
    Node node2 = Node{ 2, "B", NIL };
    Node node3 = Node{ 3, "C", NIL };
    Node node4 = Node{ 4, "D", NIL };
    // Verbindet Knoten des Graphen miteinander
    BipartiteGraph bipartiteGraph(4, 4);
    bipartiteGraph.AddEdge(&node1, &node2);
    bipartiteGraph.AddEdge(&node1, &node3);
    bipartiteGraph.AddEdge(&node2, &node1);
    bipartiteGraph.AddEdge(&node3, &node2);
    bipartiteGraph.AddEdge(&node4, &node2);
    bipartiteGraph.AddEdge(&node4, &node4);
    cout << "Size of maximum matching is " << bipartiteGraph.HopcroftKarp(); // Ausgabe auf der Konsole
}

Literatur

  • Hubertus Th. Jongen: Optimierung B. Skript zur Vorlesung. Verlag der Augustinus-Buchhandlung, Aachen, ISBN 3-925038-19-1

Einzelnachweise

  1. H. Alt, N. Blum, K. Mehlhorn, M. Paul, Institute of Computing, University of Campinas: Computing a maximum cardinality matching in a bipartite graph in time
  2. Joao C. Setubal, Institute of Computing, University of Campinas: Sequential and Parallel Experimental Results with Bipartite Matching Algorithms
  3. GeeksforGeeks: Hopcroft–Karp Algorithm for Maximum Matching
This article is issued from Wikipedia. The text is licensed under Creative Commons - Attribution - Sharealike. The authors of the article are listed here. Additional terms may apply for the media files, click on images to show image meta data.