Algoritmo de Emparejamiento de EdmondsEl Algoritmo del Blossom o Algoritmo de Emparejamiento de Edmonds es un algoritmo de teoría de grafos para construir emparejamientos máximos en grafos. El algoritmo fue desarrollado por Jack Edmonds en 1961,[1] y publicado en 1965.[2] El emparejamiento máximo es construido iterativamente mejorando el emparejamiento actual a través de caminos m-incrementos mientras al menos exista uno. La idea esencial del algoritmo es que un ciclo de longitud impar (blossom) es contraído en un solo vértice para luego continuar la búsqueda de caminos m-incrementos en el grafo resultante. La idea de contraer los ciclos de longitud impar se debe a que si no se hiciera el mismo algoritmo de búsqueda de caminos m-incrementos al entrar en uno de estos ciclos y salir pudiera reportar falsos positivos. La importancia del algoritmo radicó en que dio la primera prueba de que un emparejamiento máximo puede ser encontrado en tiempo polinomial. DefinicionesSe mantienen los nombres de las definiciones en inglés para mayor entendimiento con la literatura. Si se hiciera una traducción literal entonces 'blossom' sería capullo, 'stem' sería tallo y 'flower' sería flor. Edmonds realiza una analogía entre la estructura de las flores y las estructuras utilizadas por el algoritmo. Vértice saturadoUn vértice se dice saturado por un emparejamiento, cuando él es extremo de alguna arista del emparejamiento. BlossomUn blossom sobre un emparejamiento en un grafo , es un ciclo de longitud impar que cumple que es un emparejamiento máximo en . (Obsérvese deja uno y sólo un vértice sin estar emparejado). StemUn stem sobre un emparejamiento un grafo es un vértice no saturado por o un camino m-alternativo con un extremo no saturado y el otro sí. FlowerSe define como flower a un blossom y un stem interceptados en solo uno de los extremos del stem (vértice no saturado del blossom). Constracción de un blossom en un grafoSea un blossom de una flower sobre un emparejamiento en un grafo se define como contracción de en el grafo , al grafo que resulta de sustituir a por un vértice ficticio , el cual es adyacente a todos los vértices adyacentes a los vértices de y se denota como . Contracción de un blossom dado un emparejamientoSea un blossom de una flower sobre un emparejamiento en un grafo se define como contracción de en el emparejamiento , al emparejamiento ( sería el vértice ficticio del blossom). Se denote . TeoremaSea un blossom de una flower sobre un emparejamiento en un grafo entonces: es un emparejamiento máximo en si y solo si es un emparejamiento en . DescontracciónLa parte de la demostración más interesante del algoritmo es probar que si tiene un camino m-incremento entonces tiene un camino m-incremento semejante a . Para la demostración se separan varios casos:
AlgoritmoEl algoritmo para encontrar caminos m-incrementos utiliza una estructura de datos llamada bosque, cuyos árboles individuales corresponden a partes específicas del grafo. En cada iteración del algoritmo ocurre algunas de las siguientes opciones (1) encuentra un camino m-incremento, (2) encuentra un blossom, lo comprime y comienza la búsqueda en el grafo resultante o (3) concluye al no encontrar caminos m-incrementos. 01 Path FindAugmentingPath (Graph G, Match M) { 02 Forest F = new Forest(); // Crea un bosque vacío 03 Queue vertices = new Queue(); // Cola de procesado de vértices 04 for each (v∈V no saturado por M){ 05 F.NewTree(new Tree(v)); // Se agrega nuevo árbol con raíz v al bosque 06 vertices.enqueue(v); 07 } 08 Vertex v = vertices.dequeue(); 09 while (v != NULL ){ // distancia a la raíz par 10 Queue adjacents = new Queue(); // Cola de procesado de adyacentes. 11 for each ({v,w}∈E && {v,w}∉M} 12 adjacents.enqueue(w); 13 w = adjacents.dequeue(); 14 while (w != NULL) 15 if (w∉F){ // Si w está emparejado 16 F[root(v)].AddEdges({v,w}, {w,x}); // x emparejado con w en M 17 vertices.enqueue(x); 18 } 19 else 20 if (distance(w, root(w)) % 2 = 0){ // distancia par a su raíz 21 if (root(v) ≠ root(w)) // camino m-incremento. 22 return new Path(root(v),…, v, w, …root(w)); 23 else {// encontramos un blossom 24 Blossom B = DetectBlossom({v,w}); 25 Graph G’ = G/B; 26 Match M’ = M/B; 27 Path P’, P; 28 P’ = FindAugmentingPath(G’, M’); 29 if (P’ != NULL) 30 P = FixContractedPath(G,B,P’); 31 return P; 32 } 33 w = adjacents.dequeue(); 34 } 35 } 36 vertices.dequeue(); 37 } 38 return NULL; 39 } Coste computacionalEl ciclo de la línea 04 agrega los vértices no saturados para un costo O(|V|), el ciclo de la línea 14 se ejecuta |E| veces y por cada iteración suponiendo que se encuentra un blossom este se contrae en O(|V|) y se llama recursivamente al mismo algoritmo, partiendo de que pueden haber a lo sumo blossoms obtenemos un costo de .
Referencias
|