// Алгоритм Джонсона с использованием алгоритма Дейкстры на самоорганизующейся куче void Djonson_SelfOrganizationHeap(int** dist, Graph ADJ, int num_vert) { Graph AdditionalGraph = ADJ; AdditionalGraph.AddVertex(ADJ.GetNumOfVertex() + 1); int* dist_Bellman_Ford = new int[AdditionalGraph.GetNumOfVertex()];// Он будет содержать значения не больше нуля for(int i = 0; i < AdditionalGraph.GetNumOfVertex() - 1; i++) AdditionalGraph.AddEdge(AdditionalGraph.GetNumOfVertex() - 1, i, 0); // Применяем Беллмана-Форда для графа с наличием отрицательных весов ребер if(Bellman_Ford(dist_Bellman_Ford, AdditionalGraph, AdditionalGraph.GetNumOfVertex(), AdditionalGraph.GetNumOfVertex() - 1)) { for(ADJ.Reset(); !ADJ.IsEnd(); ADJ.GoNext())// Делаем неотрицательные веса для алгоритма Дейкстры { ADJ.SetWeightOfEdge(ADJ.GetCurrVertex(), ADJ.GetCurrEdge().vertB, ADJ.GetCurrEdge().weight + dist_Bellman_Ford[ADJ.GetCurrVertex()] - dist_Bellman_Ford[ADJ.GetCurrEdge().vertB]); } for(int i = 0; i < ADJ.GetNumOfVertex(); i++) { Dijkstra_SelfOrganizationHeap(dist[i], ADJ, ADJ.GetNumOfVertex(), i); for(int j = 0; j < ADJ.GetNumOfVertex(); j++) dist[i][j] = dist[i][j] + dist_Bellman_Ford[j] - dist_Bellman_Ford[i]; } for(ADJ.Reset(); !ADJ.IsEnd(); ADJ.GoNext())// Обратно меняем веса на предыдущие у графа ADJ { ADJ.SetWeightOfEdge(ADJ.GetCurrVertex(), ADJ.GetCurrEdge().vertB, ADJ.GetCurrEdge().weight + dist_Bellman_Ford[ADJ.GetCurrEdge().vertB ] - dist_Bellman_Ford[ADJ.GetCurrVertex()]); } } else cout << "Graph contains a cycle of negative weight" << endl; }