2020-06-25 06:43:49 +08:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* \file
|
|
|
|
* \brief [Connected Components
|
|
|
|
* (Connected Components)](https://en.wikipedia.org/wiki/Component_(graph_theory))
|
|
|
|
*
|
|
|
|
* \author [Ayaan Khan](http://github.com/ayaankhan98)
|
|
|
|
*
|
|
|
|
* \details
|
|
|
|
* A graph is a collection of nodes also called vertices and these vertices
|
|
|
|
* are connected by edges. A connected component in a graph refers to a set of
|
|
|
|
* vertices which are reachable form one another.
|
|
|
|
*
|
|
|
|
* Example - Here is graph with 3 connected components
|
|
|
|
*
|
|
|
|
* 3 9 6 8
|
|
|
|
* / \ / / \ / \
|
|
|
|
* 2---4 2 7 3 7
|
|
|
|
*
|
|
|
|
* first second third
|
|
|
|
* component component component
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <algorithm>
|
2020-05-18 18:44:51 +08:00
|
|
|
#include <iostream>
|
|
|
|
#include <vector>
|
|
|
|
|
2020-05-18 19:18:26 +08:00
|
|
|
using std::vector;
|
2020-05-18 18:44:51 +08:00
|
|
|
|
2020-06-25 06:43:49 +08:00
|
|
|
/**
|
|
|
|
* Class for representing graph as a adjacency list.
|
|
|
|
*/
|
2020-05-18 19:18:26 +08:00
|
|
|
class graph {
|
2020-05-18 19:41:15 +08:00
|
|
|
private:
|
2020-06-25 06:43:49 +08:00
|
|
|
/** \brief adj stores adjacency list representation of graph */
|
2020-05-18 19:41:15 +08:00
|
|
|
vector<vector<int>> adj;
|
2020-06-25 06:43:49 +08:00
|
|
|
|
|
|
|
/** \brief keep track of connected components */
|
2020-05-18 19:41:15 +08:00
|
|
|
int connected_components;
|
2020-06-25 06:43:49 +08:00
|
|
|
|
|
|
|
/** \brief Utility function to perform depth first search on graph */
|
2020-05-18 19:41:15 +08:00
|
|
|
void depth_first_search();
|
2020-06-25 06:43:49 +08:00
|
|
|
|
|
|
|
/** \brief Utility function that explores given vertex in graph */
|
2020-06-20 00:04:56 +08:00
|
|
|
void explore(int, vector<bool> &);
|
|
|
|
|
2020-05-18 19:41:15 +08:00
|
|
|
public:
|
2020-06-25 06:43:49 +08:00
|
|
|
/**
|
|
|
|
* \brief Constructor that intiliazes the graph on creation and set
|
|
|
|
* the connected components to 0
|
|
|
|
*/
|
2020-06-20 00:04:56 +08:00
|
|
|
explicit graph(int n) : adj(n, vector<int>()) { connected_components = 0; }
|
2020-06-25 06:43:49 +08:00
|
|
|
|
|
|
|
/** \brief Function to add a edge between two nodes or vertices of graph */
|
2020-05-18 19:41:15 +08:00
|
|
|
void addEdge(int, int);
|
2020-06-25 06:43:49 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Function the calculates the connected compoents in the graph
|
|
|
|
* by performing the depth first search on graph
|
|
|
|
*
|
|
|
|
* @return connected_components total connected components in graph
|
|
|
|
*/
|
2020-05-18 19:41:15 +08:00
|
|
|
int getConnectedComponents() {
|
2020-06-20 00:04:56 +08:00
|
|
|
depth_first_search();
|
|
|
|
return connected_components;
|
2020-05-18 19:41:15 +08:00
|
|
|
}
|
2020-05-18 18:44:51 +08:00
|
|
|
};
|
|
|
|
|
2020-06-25 06:43:49 +08:00
|
|
|
/**
|
|
|
|
* \brief Function that add edge between two nodes or vertices of graph
|
|
|
|
*
|
|
|
|
* @param u any node or vertex of graph
|
|
|
|
* @param v any node or vertex of graph
|
|
|
|
*/
|
2020-05-18 19:18:26 +08:00
|
|
|
void graph::addEdge(int u, int v) {
|
2020-06-20 00:04:56 +08:00
|
|
|
adj[u - 1].push_back(v - 1);
|
|
|
|
adj[v - 1].push_back(u - 1);
|
2020-05-18 18:44:51 +08:00
|
|
|
}
|
|
|
|
|
2020-06-25 06:43:49 +08:00
|
|
|
/**
|
|
|
|
* \brief Function that perfoms depth first search algorithm on graph
|
|
|
|
*/
|
2020-05-18 19:18:26 +08:00
|
|
|
void graph::depth_first_search() {
|
2020-06-20 00:04:56 +08:00
|
|
|
int n = adj.size();
|
|
|
|
vector<bool> visited(n, false);
|
2020-05-18 18:44:51 +08:00
|
|
|
|
2020-06-20 00:04:56 +08:00
|
|
|
for (int i = 0; i < n; i++) {
|
|
|
|
if (!visited[i]) {
|
|
|
|
explore(i, visited);
|
|
|
|
connected_components++;
|
|
|
|
}
|
2020-05-18 18:44:51 +08:00
|
|
|
}
|
|
|
|
}
|
2020-06-25 06:43:49 +08:00
|
|
|
/**
|
|
|
|
* \brief Utility function for depth first seach algorithm
|
|
|
|
* this function explores the vertex which is passed into.
|
|
|
|
*
|
|
|
|
* @param u vertex or node to be explored
|
|
|
|
* @param visited already visited vertex
|
|
|
|
*/
|
2020-05-18 19:18:26 +08:00
|
|
|
void graph::explore(int u, vector<bool> &visited) {
|
2020-06-20 00:04:56 +08:00
|
|
|
visited[u] = true;
|
|
|
|
for (auto v : adj[u]) {
|
|
|
|
if (!visited[v]) {
|
|
|
|
explore(v, visited);
|
|
|
|
}
|
2020-05-18 19:18:26 +08:00
|
|
|
}
|
2020-05-18 18:44:51 +08:00
|
|
|
}
|
|
|
|
|
2020-06-25 06:43:49 +08:00
|
|
|
/** Main function */
|
2020-05-18 19:18:26 +08:00
|
|
|
int main() {
|
2020-06-25 06:43:49 +08:00
|
|
|
/// creating a graph with 4 vertex
|
2020-06-20 00:04:56 +08:00
|
|
|
graph g(4);
|
2020-06-25 06:43:49 +08:00
|
|
|
|
|
|
|
/// Adding edges between vertices
|
2020-06-20 00:04:56 +08:00
|
|
|
g.addEdge(1, 2);
|
|
|
|
g.addEdge(3, 2);
|
2020-06-25 06:43:49 +08:00
|
|
|
|
|
|
|
/// printing the connected components
|
2020-06-20 00:04:56 +08:00
|
|
|
std::cout << g.getConnectedComponents();
|
|
|
|
return 0;
|
2020-05-18 18:44:51 +08:00
|
|
|
}
|