vcpkg/toolsrc/include/vcpkg_Graphs.h

128 lines
3.9 KiB
C
Raw Normal View History

2016-09-18 20:50:08 -07:00
#pragma once
#include <unordered_map>
2016-11-15 11:56:46 -08:00
#include <unordered_set>
2016-09-18 20:50:08 -07:00
2017-01-05 12:47:08 -08:00
namespace vcpkg::Graphs
2016-09-18 20:50:08 -07:00
{
enum class ExplorationStatus
{
// We have not visited this vertex
NOT_EXPLORED,
// We have visited this vertex but haven't visited all vertices in its subtree
PARTIALLY_EXPLORED,
// We have visited this vertex and all vertices in its subtree
FULLY_EXPLORED
};
template <class V>
class Graph
{
template <class Func>
2017-04-11 14:30:49 -07:00
static void topological_sort_internal(V vertex,
2016-09-18 20:50:08 -07:00
ExplorationStatus& status,
const Func adjacency_list_provider,
2016-09-18 20:50:08 -07:00
std::unordered_map<V, ExplorationStatus>& exploration_status,
std::vector<V>& sorted)
{
status = ExplorationStatus::PARTIALLY_EXPLORED;
auto neighbours = adjacency_list_provider(vertex);
for (V neighbour : neighbours)
2016-09-18 20:50:08 -07:00
{
ExplorationStatus& neighbour_status = exploration_status[neighbour];
if (neighbour_status == ExplorationStatus::NOT_EXPLORED)
{
2017-04-11 14:30:49 -07:00
topological_sort_internal(neighbour, neighbour_status, adjacency_list_provider, exploration_status, sorted);
2016-09-18 20:50:08 -07:00
}
else if (neighbour_status == ExplorationStatus::PARTIALLY_EXPLORED)
{
throw std::runtime_error("cycle in graph");
}
}
status = ExplorationStatus::FULLY_EXPLORED;
sorted.push_back(vertex);
}
public:
void add_vertex(V v)
{
this->vertices[v];
}
// TODO: Change with iterators
void add_vertices(const std::vector<V>& vs)
{
for (const V& v : vs)
{
this->vertices[v];
}
}
void add_edge(V u, V v)
{
this->vertices[v];
2016-11-15 11:56:46 -08:00
this->vertices[u].insert(v);
2016-09-18 20:50:08 -07:00
}
2017-04-11 14:30:49 -07:00
std::vector<V> topological_sort() const
2016-09-18 20:50:08 -07:00
{
std::unordered_map<V, int> indegrees = count_indegrees();
std::vector<V> sorted;
sorted.reserve(indegrees.size());
std::unordered_map<V, ExplorationStatus> exploration_status;
exploration_status.reserve(indegrees.size());
for (auto& pair : indegrees)
{
if (pair.second == 0) // Starting from vertices with indegree == 0. Not required.
{
V vertex = pair.first;
ExplorationStatus& status = exploration_status[vertex];
if (status == ExplorationStatus::NOT_EXPLORED)
{
2017-04-11 14:30:49 -07:00
topological_sort_internal(vertex,
status,
[this](const V& v) { return this->vertices.at(v); },
exploration_status,
sorted);
2016-09-18 20:50:08 -07:00
}
}
}
return sorted;
}
std::unordered_map<V, int> count_indegrees() const
{
std::unordered_map<V, int> indegrees;
for (auto& pair : this->vertices)
{
indegrees[pair.first];
for (V neighbour : pair.second)
{
++indegrees[neighbour];
}
}
return indegrees;
}
2016-11-15 11:56:46 -08:00
const std::unordered_map<V, std::unordered_set<V>>& adjacency_list() const
2016-09-18 20:50:08 -07:00
{
return this->vertices;
}
private:
2016-11-15 11:56:46 -08:00
std::unordered_map<V, std::unordered_set<V>> vertices;
2016-09-18 20:50:08 -07:00
};
2017-01-05 12:47:08 -08:00
}